@juspay/shooter 1.17.0 → 1.19.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 (342) hide show
  1. package/build/client/_app/immutable/assets/{0.B0O0vCnX.css → 0.BwNtE8TX.css} +1 -1
  2. package/build/client/_app/immutable/assets/0.BwNtE8TX.css.br +0 -0
  3. package/build/client/_app/immutable/assets/{0.B0O0vCnX.css.gz → 0.BwNtE8TX.css.gz} +0 -0
  4. package/build/client/_app/immutable/assets/8.BYgAX7hR.css +1 -0
  5. package/build/client/_app/immutable/assets/8.BYgAX7hR.css.br +0 -0
  6. package/build/client/_app/immutable/assets/8.BYgAX7hR.css.gz +0 -0
  7. package/build/client/_app/immutable/assets/9.DV6pZunn.css +1 -0
  8. package/build/client/_app/immutable/assets/9.DV6pZunn.css.br +0 -0
  9. package/build/client/_app/immutable/assets/9.DV6pZunn.css.gz +0 -0
  10. package/build/client/_app/immutable/chunks/{DZQMsHM5.js → 2rBV5OkJ.js} +1 -1
  11. package/build/client/_app/immutable/chunks/2rBV5OkJ.js.br +0 -0
  12. package/build/client/_app/immutable/chunks/2rBV5OkJ.js.gz +0 -0
  13. package/build/client/_app/immutable/chunks/BB2l8o4X.js +1 -0
  14. package/build/client/_app/immutable/chunks/BB2l8o4X.js.br +0 -0
  15. package/build/client/_app/immutable/chunks/BB2l8o4X.js.gz +0 -0
  16. package/build/client/_app/immutable/chunks/{Cg3dlX05.js → BPDiEZo0.js} +2 -2
  17. package/build/client/_app/immutable/chunks/BPDiEZo0.js.br +0 -0
  18. package/build/client/_app/immutable/chunks/BPDiEZo0.js.gz +0 -0
  19. package/build/client/_app/immutable/chunks/{C_9BZILB.js → BcpydfqI.js} +1 -1
  20. package/build/client/_app/immutable/chunks/BcpydfqI.js.br +0 -0
  21. package/build/client/_app/immutable/chunks/BcpydfqI.js.gz +0 -0
  22. package/build/client/_app/immutable/chunks/BcqA7eKM.js +3 -0
  23. package/build/client/_app/immutable/chunks/BcqA7eKM.js.br +0 -0
  24. package/build/client/_app/immutable/chunks/BcqA7eKM.js.gz +0 -0
  25. package/build/client/_app/immutable/chunks/BdtLzPpO.js +1 -0
  26. package/build/client/_app/immutable/chunks/BdtLzPpO.js.br +0 -0
  27. package/build/client/_app/immutable/chunks/BdtLzPpO.js.gz +0 -0
  28. package/build/client/_app/immutable/chunks/{BRqaaL5D.js → BvmdJful.js} +1 -1
  29. package/build/client/_app/immutable/chunks/BvmdJful.js.br +0 -0
  30. package/build/client/_app/immutable/chunks/BvmdJful.js.gz +0 -0
  31. package/build/client/_app/immutable/chunks/{C5VOyQCG.js → ClIPTXf3.js} +1 -1
  32. package/build/client/_app/immutable/chunks/ClIPTXf3.js.br +0 -0
  33. package/build/client/_app/immutable/chunks/ClIPTXf3.js.gz +0 -0
  34. package/build/client/_app/immutable/chunks/{BctvtE4d.js → DA4Zt9Me.js} +1 -1
  35. package/build/client/_app/immutable/chunks/DA4Zt9Me.js.br +0 -0
  36. package/build/client/_app/immutable/chunks/{BctvtE4d.js.gz → DA4Zt9Me.js.gz} +0 -0
  37. package/build/client/_app/immutable/chunks/{CjfxuHdN.js → DCDL_9ys.js} +1 -1
  38. package/build/client/_app/immutable/chunks/DCDL_9ys.js.br +0 -0
  39. package/build/client/_app/immutable/chunks/DCDL_9ys.js.gz +0 -0
  40. package/build/client/_app/immutable/chunks/{DYuMZGL5.js → DWmC0QM7.js} +1 -1
  41. package/build/client/_app/immutable/chunks/DWmC0QM7.js.br +0 -0
  42. package/build/client/_app/immutable/chunks/DWmC0QM7.js.gz +0 -0
  43. package/build/client/_app/immutable/chunks/{DZvnhU_8.js → ZS5XYDx_.js} +2 -2
  44. package/build/client/_app/immutable/chunks/ZS5XYDx_.js.br +0 -0
  45. package/build/client/_app/immutable/chunks/ZS5XYDx_.js.gz +0 -0
  46. package/build/client/_app/immutable/entry/app.D4TXlu7A.js +2 -0
  47. package/build/client/_app/immutable/entry/app.D4TXlu7A.js.br +0 -0
  48. package/build/client/_app/immutable/entry/app.D4TXlu7A.js.gz +0 -0
  49. package/build/client/_app/immutable/entry/start.BBQhtURO.js +1 -0
  50. package/build/client/_app/immutable/entry/start.BBQhtURO.js.br +0 -0
  51. package/build/client/_app/immutable/entry/start.BBQhtURO.js.gz +0 -0
  52. package/build/client/_app/immutable/nodes/0.1zylwAPT.js +10 -0
  53. package/build/client/_app/immutable/nodes/0.1zylwAPT.js.br +0 -0
  54. package/build/client/_app/immutable/nodes/0.1zylwAPT.js.gz +0 -0
  55. package/build/client/_app/immutable/nodes/{1.Fqso94b3.js → 1.BVnLUSs-.js} +1 -1
  56. package/build/client/_app/immutable/nodes/1.BVnLUSs-.js.br +0 -0
  57. package/build/client/_app/immutable/nodes/1.BVnLUSs-.js.gz +0 -0
  58. package/build/client/_app/immutable/nodes/{8.BjKgvSie.js → 10.D1wl2wPX.js} +2 -2
  59. package/build/client/_app/immutable/nodes/10.D1wl2wPX.js.br +0 -0
  60. package/build/client/_app/immutable/nodes/10.D1wl2wPX.js.gz +0 -0
  61. package/build/client/_app/immutable/nodes/{9.BRT6HOXB.js → 11.C18nMGmp.js} +1 -1
  62. package/build/client/_app/immutable/nodes/11.C18nMGmp.js.br +0 -0
  63. package/build/client/_app/immutable/nodes/11.C18nMGmp.js.gz +0 -0
  64. package/build/client/_app/immutable/nodes/{2.BusCVJWk.js → 2.D1Mm0DUX.js} +2 -2
  65. package/build/client/_app/immutable/nodes/2.D1Mm0DUX.js.br +0 -0
  66. package/build/client/_app/immutable/nodes/2.D1Mm0DUX.js.gz +0 -0
  67. package/build/client/_app/immutable/nodes/{3.DUlpocIc.js → 3.Wfz3TcJd.js} +3 -3
  68. package/build/client/_app/immutable/nodes/3.Wfz3TcJd.js.br +0 -0
  69. package/build/client/_app/immutable/nodes/3.Wfz3TcJd.js.gz +0 -0
  70. package/build/client/_app/immutable/nodes/{4.BSVqdrrD.js → 4.CBX9A3ka.js} +2 -2
  71. package/build/client/_app/immutable/nodes/4.CBX9A3ka.js.br +0 -0
  72. package/build/client/_app/immutable/nodes/4.CBX9A3ka.js.gz +0 -0
  73. package/build/client/_app/immutable/nodes/{5.Cfj35gpY.js → 5.DIVKuZc9.js} +1 -1
  74. package/build/client/_app/immutable/nodes/5.DIVKuZc9.js.br +0 -0
  75. package/build/client/_app/immutable/nodes/5.DIVKuZc9.js.gz +0 -0
  76. package/build/client/_app/immutable/nodes/{6.CG4eKRH0.js → 6.DtZAEPXb.js} +1 -1
  77. package/build/client/_app/immutable/nodes/6.DtZAEPXb.js.br +0 -0
  78. package/build/client/_app/immutable/nodes/6.DtZAEPXb.js.gz +0 -0
  79. package/build/client/_app/immutable/nodes/{7.DHilxD1o.js → 7.MfBRh32I.js} +1 -1
  80. package/build/client/_app/immutable/nodes/7.MfBRh32I.js.br +0 -0
  81. package/build/client/_app/immutable/nodes/7.MfBRh32I.js.gz +0 -0
  82. package/build/client/_app/immutable/nodes/8.DVE6LnOC.js +1 -0
  83. package/build/client/_app/immutable/nodes/8.DVE6LnOC.js.br +0 -0
  84. package/build/client/_app/immutable/nodes/8.DVE6LnOC.js.gz +0 -0
  85. package/build/client/_app/immutable/nodes/9.BCel5OqI.js +2 -0
  86. package/build/client/_app/immutable/nodes/9.BCel5OqI.js.br +0 -0
  87. package/build/client/_app/immutable/nodes/9.BCel5OqI.js.gz +0 -0
  88. package/build/client/_app/version.json +1 -1
  89. package/build/client/_app/version.json.br +0 -0
  90. package/build/client/_app/version.json.gz +0 -0
  91. package/build/server/chunks/{0-BWFSL107.js → 0-DJqyZZTr.js} +4 -4
  92. package/build/server/chunks/{0-BWFSL107.js.map → 0-DJqyZZTr.js.map} +1 -1
  93. package/build/server/chunks/1-2YUVen1F.js +9 -0
  94. package/build/server/chunks/{1-Bw5KlAjL.js.map → 1-2YUVen1F.js.map} +1 -1
  95. package/build/server/chunks/10-D1X7LB3v.js +9 -0
  96. package/build/server/chunks/10-D1X7LB3v.js.map +1 -0
  97. package/build/server/chunks/11-qXSPdF5j.js +9 -0
  98. package/build/server/chunks/11-qXSPdF5j.js.map +1 -0
  99. package/build/server/chunks/{2-CQ3yYSVK.js → 2-BD7kj1mt.js} +3 -3
  100. package/build/server/chunks/{2-CQ3yYSVK.js.map → 2-BD7kj1mt.js.map} +1 -1
  101. package/build/server/chunks/{3-DZ4H9hPs.js → 3-oNjv-BhZ.js} +3 -3
  102. package/build/server/chunks/{3-DZ4H9hPs.js.map → 3-oNjv-BhZ.js.map} +1 -1
  103. package/build/server/chunks/{4-BtYdKCVW.js → 4-Bb5VFhsO.js} +3 -3
  104. package/build/server/chunks/{4-BtYdKCVW.js.map → 4-Bb5VFhsO.js.map} +1 -1
  105. package/build/server/chunks/{5-CvJK3PiH.js → 5-oNoWuIsn.js} +3 -3
  106. package/build/server/chunks/{5-CvJK3PiH.js.map → 5-oNoWuIsn.js.map} +1 -1
  107. package/build/server/chunks/6-DRJGUqHG.js +9 -0
  108. package/build/server/chunks/{6-BZ0enR6b.js.map → 6-DRJGUqHG.js.map} +1 -1
  109. package/build/server/chunks/7-_giJiu0L.js +9 -0
  110. package/build/server/chunks/{7-Lg8imTZn.js.map → 7-_giJiu0L.js.map} +1 -1
  111. package/build/server/chunks/8-zvWAVNT5.js +9 -0
  112. package/build/server/chunks/8-zvWAVNT5.js.map +1 -0
  113. package/build/server/chunks/9-DVyDL445.js +9 -0
  114. package/build/server/chunks/9-DVyDL445.js.map +1 -0
  115. package/build/server/chunks/Banner-BgaAs1rs.js +90 -0
  116. package/build/server/chunks/Banner-BgaAs1rs.js.map +1 -0
  117. package/build/server/chunks/{Button-B5dU-ntz.js → Button-D0hZ7JYt.js} +2 -2
  118. package/build/server/chunks/Button-D0hZ7JYt.js.map +1 -0
  119. package/build/server/chunks/{Icon-C7Ml3GX6.js → Icon-D0GBnDcs.js} +3 -3
  120. package/build/server/chunks/Icon-D0GBnDcs.js.map +1 -0
  121. package/build/server/chunks/{Input-CPGO0sbS.js → Input-OmIiydSx.js} +2 -2
  122. package/build/server/chunks/Input-OmIiydSx.js.map +1 -0
  123. package/build/server/chunks/{Pill-CcrtCejm.js → Pill-4xJ-VhAA.js} +3 -3
  124. package/build/server/chunks/Pill-4xJ-VhAA.js.map +1 -0
  125. package/build/server/chunks/{Shimmer-C5jkvGr1.js → Shimmer-Dw2uvTC1.js} +2 -2
  126. package/build/server/chunks/Shimmer-Dw2uvTC1.js.map +1 -0
  127. package/build/server/chunks/{_error.svelte-CSIxs-ab.js → _error.svelte-CZnkxeLr.js} +8 -8
  128. package/build/server/chunks/{_error.svelte-CSIxs-ab.js.map → _error.svelte-CZnkxeLr.js.map} +1 -1
  129. package/build/server/chunks/{_layout.svelte-noB4j-v2.js → _layout.svelte-DfgNGGiM.js} +16 -11
  130. package/build/server/chunks/_layout.svelte-DfgNGGiM.js.map +1 -0
  131. package/build/server/chunks/{_page.svelte-DnTpPnPR.js → _page.svelte-BLo2v_8E.js} +7 -88
  132. package/build/server/chunks/_page.svelte-BLo2v_8E.js.map +1 -0
  133. package/build/server/chunks/_page.svelte-BTlfUsBp.js +43 -0
  134. package/build/server/chunks/_page.svelte-BTlfUsBp.js.map +1 -0
  135. package/build/server/chunks/{_page.svelte-C60lAagP.js → _page.svelte-BUBLUSGo.js} +8 -8
  136. package/build/server/chunks/_page.svelte-BUBLUSGo.js.map +1 -0
  137. package/build/server/chunks/{_page.svelte-BV0XyYJZ.js → _page.svelte-BX2FMgSg.js} +4 -4
  138. package/build/server/chunks/{_page.svelte-BV0XyYJZ.js.map → _page.svelte-BX2FMgSg.js.map} +1 -1
  139. package/build/server/chunks/{_page.svelte-BUkm2304.js → _page.svelte-C7B0qdrC.js} +5 -5
  140. package/build/server/chunks/{_page.svelte-BUkm2304.js.map → _page.svelte-C7B0qdrC.js.map} +1 -1
  141. package/build/server/chunks/{_page.svelte-Dmg-RFCg.js → _page.svelte-CE7COWnF.js} +7 -7
  142. package/build/server/chunks/{_page.svelte-Dmg-RFCg.js.map → _page.svelte-CE7COWnF.js.map} +1 -1
  143. package/build/server/chunks/{_page.svelte-BfB8maoc.js → _page.svelte-CWsjjd4l.js} +9 -9
  144. package/build/server/chunks/{_page.svelte-BfB8maoc.js.map → _page.svelte-CWsjjd4l.js.map} +1 -1
  145. package/build/server/chunks/_page.svelte-D5S2hkBk.js +104 -0
  146. package/build/server/chunks/_page.svelte-D5S2hkBk.js.map +1 -0
  147. package/build/server/chunks/{_page.svelte-B6qyh-K-.js → _page.svelte-D_Ey8QRG.js} +11 -11
  148. package/build/server/chunks/{_page.svelte-B6qyh-K-.js.map → _page.svelte-D_Ey8QRG.js.map} +1 -1
  149. package/build/server/chunks/{_page.svelte-DuzZr5dA.js → _page.svelte-tBuIq8Pg.js} +11 -11
  150. package/build/server/chunks/{_page.svelte-DuzZr5dA.js.map → _page.svelte-tBuIq8Pg.js.map} +1 -1
  151. package/build/server/chunks/_server.ts-BaaY7Z9D.js +77 -0
  152. package/build/server/chunks/_server.ts-BaaY7Z9D.js.map +1 -0
  153. package/build/server/chunks/{_server.ts-5wx4ZppI.js → _server.ts-Bi0Oe4PF.js} +7 -4
  154. package/build/server/chunks/_server.ts-Bi0Oe4PF.js.map +1 -0
  155. package/build/server/chunks/_server.ts-C0317RBD.js +57 -0
  156. package/build/server/chunks/_server.ts-C0317RBD.js.map +1 -0
  157. package/build/server/chunks/{_server.ts-CKXVBbwb.js → _server.ts-CRVNEOd2.js} +10 -8
  158. package/build/server/chunks/_server.ts-CRVNEOd2.js.map +1 -0
  159. package/build/server/chunks/_server.ts-CVPZOpiv.js +23 -0
  160. package/build/server/chunks/_server.ts-CVPZOpiv.js.map +1 -0
  161. package/build/server/chunks/{_server.ts-CyjDrcZN.js → _server.ts-C_OOUqsd.js} +9 -1
  162. package/build/server/chunks/_server.ts-C_OOUqsd.js.map +1 -0
  163. package/build/server/chunks/{_server.ts-BMMTS86y.js → _server.ts-D9ir7u24.js} +3 -4
  164. package/build/server/chunks/{_server.ts-BMMTS86y.js.map → _server.ts-D9ir7u24.js.map} +1 -1
  165. package/build/server/chunks/{_server.ts-DZ5naqSL.js → _server.ts-DMm0hBP4.js} +5 -1
  166. package/build/server/chunks/_server.ts-DMm0hBP4.js.map +1 -0
  167. package/build/server/chunks/{_server.ts-CgHc1Zpx.js → _server.ts-DhJx0DLr.js} +7 -4
  168. package/build/server/chunks/_server.ts-DhJx0DLr.js.map +1 -0
  169. package/build/server/chunks/_server.ts-DkZX_O9a.js +39 -0
  170. package/build/server/chunks/_server.ts-DkZX_O9a.js.map +1 -0
  171. package/build/server/chunks/{_server.ts-B1z0q6qZ.js → _server.ts-DxT9IlZF.js} +6 -5
  172. package/build/server/chunks/_server.ts-DxT9IlZF.js.map +1 -0
  173. package/build/server/chunks/{_server.ts-Bt7EAfjo.js → _server.ts-MbnroWEF.js} +25 -48
  174. package/build/server/chunks/_server.ts-MbnroWEF.js.map +1 -0
  175. package/build/server/chunks/_server.ts-Mttr0-Sl.js +48 -0
  176. package/build/server/chunks/_server.ts-Mttr0-Sl.js.map +1 -0
  177. package/build/server/chunks/_server.ts-jtqWDWcf.js +45 -0
  178. package/build/server/chunks/_server.ts-jtqWDWcf.js.map +1 -0
  179. package/build/server/chunks/{cache-Me3zUAaD.js → cache-BlMaDsHi.js} +2 -2
  180. package/build/server/chunks/{cache-Me3zUAaD.js.map → cache-BlMaDsHi.js.map} +1 -1
  181. package/build/server/chunks/{client-CfNnl32g.js → client-Ds1brw-8.js} +4 -4
  182. package/build/server/chunks/{client-CfNnl32g.js.map → client-Ds1brw-8.js.map} +1 -1
  183. package/build/server/chunks/client2-DngLdcUc.js +7 -0
  184. package/build/server/chunks/{client2-DDP30_vY.js.map → client2-DngLdcUc.js.map} +1 -1
  185. package/build/server/chunks/coordinator-DMU_ADXf.js +530 -0
  186. package/build/server/chunks/coordinator-DMU_ADXf.js.map +1 -0
  187. package/build/server/chunks/{index-CJrGuxuM.js → index-CoYB03g7.js} +2 -2
  188. package/build/server/chunks/{index-CJrGuxuM.js.map → index-CoYB03g7.js.map} +1 -1
  189. package/build/server/chunks/{index-server--49oHtA0.js → index-server-Bq3cnK69.js} +2 -2
  190. package/build/server/chunks/{index-server--49oHtA0.js.map → index-server-Bq3cnK69.js.map} +1 -1
  191. package/build/server/chunks/{index2-MY7PXeAc.js → index2-dSGQ9Eaa.js} +2 -2
  192. package/build/server/chunks/{index2-MY7PXeAc.js.map → index2-dSGQ9Eaa.js.map} +1 -1
  193. package/build/server/chunks/{pty-manager-RmhVe2Ez.js → pty-manager-41h3IK8K.js} +100 -2
  194. package/build/server/chunks/pty-manager-41h3IK8K.js.map +1 -0
  195. package/build/server/chunks/qwen-reader-DGfUbKaJ.js +2112 -0
  196. package/build/server/chunks/qwen-reader-DGfUbKaJ.js.map +1 -0
  197. package/build/server/chunks/{registry-DzJj2E6I.js → registry-D4J_CuzW.js} +56 -24
  198. package/build/server/chunks/registry-D4J_CuzW.js.map +1 -0
  199. package/build/server/chunks/{root-xvQIR1Bt.js → root-D4IoFC8F.js} +2 -2
  200. package/build/server/chunks/root-D4IoFC8F.js.map +1 -0
  201. package/build/server/chunks/{state.svelte-RCtlkrNH.js → state.svelte-CmHqngc_.js} +3 -3
  202. package/build/server/chunks/{state.svelte-RCtlkrNH.js.map → state.svelte-CmHqngc_.js.map} +1 -1
  203. package/build/server/chunks/{stores-C-LqoonT.js → stores-CRYxfF0o.js} +4 -4
  204. package/build/server/chunks/stores-CRYxfF0o.js.map +1 -0
  205. package/build/server/chunks/super-session-handler-DPyxFgmz.js +22 -0
  206. package/build/server/chunks/super-session-handler-DPyxFgmz.js.map +1 -0
  207. package/build/server/index.js +4 -4
  208. package/build/server/index.js.map +1 -1
  209. package/build/server/manifest.js +79 -21
  210. package/build/server/manifest.js.map +1 -1
  211. package/package.json +2 -2
  212. package/scripts/e2e-all-features.sh +204 -0
  213. package/scripts/e2e-cross-terminal.sh +168 -0
  214. package/server.ts +37 -0
  215. package/src/lib/modules/client/common/provider.ts +0 -2
  216. package/src/lib/modules/client/terminal/ChatView.svelte +9 -2
  217. package/src/lib/modules/client/terminal/LaunchSheet.svelte +3 -0
  218. package/src/lib/modules/server/sessions/amp-reader.ts +439 -0
  219. package/src/lib/modules/server/sessions/copilot-reader.ts +542 -0
  220. package/src/lib/modules/server/sessions/cursor-reader.ts +634 -0
  221. package/src/lib/modules/server/sessions/gemini-reader.ts +48 -25
  222. package/src/lib/modules/server/sessions/opencode-reader.ts +13 -12
  223. package/src/lib/modules/server/sessions/process-detector.ts +37 -60
  224. package/src/lib/modules/server/sessions/provider-paths.ts +173 -0
  225. package/src/lib/modules/server/sessions/qwen-reader.ts +41 -15
  226. package/src/lib/modules/server/sessions/registry.ts +55 -14
  227. package/src/lib/modules/server/sos/coordinator.ts +492 -0
  228. package/src/lib/modules/server/sos/policy-gate.ts +56 -0
  229. package/src/lib/modules/server/sos/relay-store.ts +159 -0
  230. package/src/lib/modules/server/terminal/generic-session-watcher.ts +163 -0
  231. package/src/lib/modules/server/terminal/pty-input.ts +37 -0
  232. package/src/lib/modules/server/terminal/pty-manager.ts +51 -0
  233. package/src/lib/modules/server/ws/server.ts +6 -1
  234. package/src/lib/modules/server/ws/session-handler.ts +17 -2
  235. package/src/lib/modules/server/ws/super-session-handler.ts +200 -0
  236. package/src/lib/theme.css +1 -2
  237. package/src/lib/types/generated/Sessions.ts +1 -4
  238. package/src/lib/types/index.ts +2 -1
  239. package/src/lib/types/server.ts +23 -6
  240. package/src/lib/types/sessions.ts +1 -10
  241. package/src/lib/types/sos.ts +134 -0
  242. package/src/routes/+layout.svelte +9 -2
  243. package/src/routes/api/sessions/connect/+server.ts +7 -3
  244. package/src/routes/api/sos/+server.ts +36 -0
  245. package/src/routes/api/sos/[id]/+server.ts +55 -0
  246. package/src/routes/api/sos/[id]/inject/+server.ts +44 -0
  247. package/src/routes/api/sos/[id]/members/+server.ts +47 -0
  248. package/src/routes/api/sos/[id]/members/[mid]/+server.ts +17 -0
  249. package/src/routes/api/sos/[id]/rules/+server.ts +85 -0
  250. package/src/routes/sos/+page.svelte +195 -0
  251. package/src/routes/sos/[id]/+page.svelte +677 -0
  252. package/build/client/_app/immutable/assets/0.B0O0vCnX.css.br +0 -0
  253. package/build/client/_app/immutable/chunks/BRqaaL5D.js.br +0 -0
  254. package/build/client/_app/immutable/chunks/BRqaaL5D.js.gz +0 -0
  255. package/build/client/_app/immutable/chunks/BctvtE4d.js.br +0 -0
  256. package/build/client/_app/immutable/chunks/BxFShcQO.js +0 -1
  257. package/build/client/_app/immutable/chunks/BxFShcQO.js.br +0 -0
  258. package/build/client/_app/immutable/chunks/BxFShcQO.js.gz +0 -0
  259. package/build/client/_app/immutable/chunks/ByzqAuXw.js +0 -3
  260. package/build/client/_app/immutable/chunks/ByzqAuXw.js.br +0 -0
  261. package/build/client/_app/immutable/chunks/ByzqAuXw.js.gz +0 -0
  262. package/build/client/_app/immutable/chunks/C5VOyQCG.js.br +0 -0
  263. package/build/client/_app/immutable/chunks/C5VOyQCG.js.gz +0 -0
  264. package/build/client/_app/immutable/chunks/C_9BZILB.js.br +0 -0
  265. package/build/client/_app/immutable/chunks/C_9BZILB.js.gz +0 -0
  266. package/build/client/_app/immutable/chunks/Cg3dlX05.js.br +0 -0
  267. package/build/client/_app/immutable/chunks/Cg3dlX05.js.gz +0 -0
  268. package/build/client/_app/immutable/chunks/CjfxuHdN.js.br +0 -0
  269. package/build/client/_app/immutable/chunks/CjfxuHdN.js.gz +0 -0
  270. package/build/client/_app/immutable/chunks/DYuMZGL5.js.br +0 -0
  271. package/build/client/_app/immutable/chunks/DYuMZGL5.js.gz +0 -0
  272. package/build/client/_app/immutable/chunks/DZQMsHM5.js.br +0 -0
  273. package/build/client/_app/immutable/chunks/DZQMsHM5.js.gz +0 -0
  274. package/build/client/_app/immutable/chunks/DZvnhU_8.js.br +0 -0
  275. package/build/client/_app/immutable/chunks/DZvnhU_8.js.gz +0 -0
  276. package/build/client/_app/immutable/chunks/Pw0jDB7M.js +0 -1
  277. package/build/client/_app/immutable/chunks/Pw0jDB7M.js.br +0 -0
  278. package/build/client/_app/immutable/chunks/Pw0jDB7M.js.gz +0 -0
  279. package/build/client/_app/immutable/entry/app.CNaTe-zm.js +0 -2
  280. package/build/client/_app/immutable/entry/app.CNaTe-zm.js.br +0 -0
  281. package/build/client/_app/immutable/entry/app.CNaTe-zm.js.gz +0 -0
  282. package/build/client/_app/immutable/entry/start.hxYnjcDu.js +0 -1
  283. package/build/client/_app/immutable/entry/start.hxYnjcDu.js.br +0 -0
  284. package/build/client/_app/immutable/entry/start.hxYnjcDu.js.gz +0 -0
  285. package/build/client/_app/immutable/nodes/0.C3ELOf4c.js +0 -7
  286. package/build/client/_app/immutable/nodes/0.C3ELOf4c.js.br +0 -0
  287. package/build/client/_app/immutable/nodes/0.C3ELOf4c.js.gz +0 -0
  288. package/build/client/_app/immutable/nodes/1.Fqso94b3.js.br +0 -0
  289. package/build/client/_app/immutable/nodes/1.Fqso94b3.js.gz +0 -0
  290. package/build/client/_app/immutable/nodes/2.BusCVJWk.js.br +0 -0
  291. package/build/client/_app/immutable/nodes/2.BusCVJWk.js.gz +0 -0
  292. package/build/client/_app/immutable/nodes/3.DUlpocIc.js.br +0 -0
  293. package/build/client/_app/immutable/nodes/3.DUlpocIc.js.gz +0 -0
  294. package/build/client/_app/immutable/nodes/4.BSVqdrrD.js.br +0 -0
  295. package/build/client/_app/immutable/nodes/4.BSVqdrrD.js.gz +0 -0
  296. package/build/client/_app/immutable/nodes/5.Cfj35gpY.js.br +0 -0
  297. package/build/client/_app/immutable/nodes/5.Cfj35gpY.js.gz +0 -0
  298. package/build/client/_app/immutable/nodes/6.CG4eKRH0.js.br +0 -0
  299. package/build/client/_app/immutable/nodes/6.CG4eKRH0.js.gz +0 -0
  300. package/build/client/_app/immutable/nodes/7.DHilxD1o.js.br +0 -0
  301. package/build/client/_app/immutable/nodes/7.DHilxD1o.js.gz +0 -0
  302. package/build/client/_app/immutable/nodes/8.BjKgvSie.js.br +0 -0
  303. package/build/client/_app/immutable/nodes/8.BjKgvSie.js.gz +0 -0
  304. package/build/client/_app/immutable/nodes/9.BRT6HOXB.js.br +0 -0
  305. package/build/client/_app/immutable/nodes/9.BRT6HOXB.js.gz +0 -0
  306. package/build/server/chunks/1-Bw5KlAjL.js +0 -9
  307. package/build/server/chunks/6-BZ0enR6b.js +0 -9
  308. package/build/server/chunks/7-Lg8imTZn.js +0 -9
  309. package/build/server/chunks/8-DKs4yOL7.js +0 -9
  310. package/build/server/chunks/8-DKs4yOL7.js.map +0 -1
  311. package/build/server/chunks/9-UNmpUWDY.js +0 -9
  312. package/build/server/chunks/9-UNmpUWDY.js.map +0 -1
  313. package/build/server/chunks/Button-B5dU-ntz.js.map +0 -1
  314. package/build/server/chunks/Icon-C7Ml3GX6.js.map +0 -1
  315. package/build/server/chunks/Input-CPGO0sbS.js.map +0 -1
  316. package/build/server/chunks/Pill-CcrtCejm.js.map +0 -1
  317. package/build/server/chunks/Shimmer-C5jkvGr1.js.map +0 -1
  318. package/build/server/chunks/_layout.svelte-noB4j-v2.js.map +0 -1
  319. package/build/server/chunks/_page.svelte-C60lAagP.js.map +0 -1
  320. package/build/server/chunks/_page.svelte-DnTpPnPR.js.map +0 -1
  321. package/build/server/chunks/_server.ts-5wx4ZppI.js.map +0 -1
  322. package/build/server/chunks/_server.ts-B1z0q6qZ.js.map +0 -1
  323. package/build/server/chunks/_server.ts-Bt7EAfjo.js.map +0 -1
  324. package/build/server/chunks/_server.ts-CKXVBbwb.js.map +0 -1
  325. package/build/server/chunks/_server.ts-CgHc1Zpx.js.map +0 -1
  326. package/build/server/chunks/_server.ts-CyjDrcZN.js.map +0 -1
  327. package/build/server/chunks/_server.ts-DZ5naqSL.js.map +0 -1
  328. package/build/server/chunks/client2-DDP30_vY.js +0 -7
  329. package/build/server/chunks/opencode-db-path-BwaPufWf.js +0 -411
  330. package/build/server/chunks/opencode-db-path-BwaPufWf.js.map +0 -1
  331. package/build/server/chunks/pty-manager-RmhVe2Ez.js.map +0 -1
  332. package/build/server/chunks/qwen-reader-2fTFuC_D.js +0 -622
  333. package/build/server/chunks/qwen-reader-2fTFuC_D.js.map +0 -1
  334. package/build/server/chunks/registry-DzJj2E6I.js.map +0 -1
  335. package/build/server/chunks/root-xvQIR1Bt.js.map +0 -1
  336. package/build/server/chunks/stores-C-LqoonT.js.map +0 -1
  337. /package/build/client/_app/immutable/assets/{8.BhoBXADL.css → 10.BhoBXADL.css} +0 -0
  338. /package/build/client/_app/immutable/assets/{8.BhoBXADL.css.br → 10.BhoBXADL.css.br} +0 -0
  339. /package/build/client/_app/immutable/assets/{8.BhoBXADL.css.gz → 10.BhoBXADL.css.gz} +0 -0
  340. /package/build/client/_app/immutable/assets/{9.v5KA95xm.css → 11.v5KA95xm.css} +0 -0
  341. /package/build/client/_app/immutable/assets/{9.v5KA95xm.css.br → 11.v5KA95xm.css.br} +0 -0
  342. /package/build/client/_app/immutable/assets/{9.v5KA95xm.css.gz → 11.v5KA95xm.css.gz} +0 -0
@@ -0,0 +1,134 @@
1
+ // Session-Over-Sessions (SoS) types — the coordinator that merges N running
2
+ // agent sessions into one source-tagged super-session and relays messages
3
+ // between them. See docs/SESSION-OVER-SESSIONS.md.
4
+ //
5
+ // A member's `sessionKey` is exactly what the server's session-watcher adapter
6
+ // keys on: a JSONL/JSON file path for file-backed providers, or a bare session
7
+ // id for OpenCode. That lets the coordinator reuse the same watcher routing the
8
+ // WS session handler uses, covering all providers with no new watching code.
9
+
10
+ import type { SessionSource } from './generated';
11
+ import type { ConversationMessage } from './sessions';
12
+
13
+ /** An escalated relay awaiting human approve/deny (Phase 2 HITL). */
14
+ export interface PendingRelay {
15
+ createdAt: string;
16
+ expiresAt: string;
17
+ fromMemberId: string;
18
+ id: string;
19
+ text: string;
20
+ timer: null | ReturnType<typeof setTimeout>;
21
+ toMemberId: string;
22
+ }
23
+
24
+ /** A control message sent from a SoS client to the coordinator over /ws/super-session/:id. */
25
+ export type SosClientMessage =
26
+ | {
27
+ capability?: string;
28
+ provider: SessionSource;
29
+ sessionKey: string;
30
+ terminalId?: string;
31
+ type: 'member-add';
32
+ }
33
+ | { memberId: string; type: 'member-remove' }
34
+ | { relayId: string; type: 'relay-approve' }
35
+ | { relayId: string; type: 'relay-deny' }
36
+ | { text: string; toMemberId: string; type: 'relay-forward' };
37
+
38
+ /**
39
+ * Injects relay text into a Shooter-owned terminal's stdin. Supplied by
40
+ * server.ts (which owns PtyManager) so the coordinator stays free of PTY deps.
41
+ * Performs the ownership check (terminal exists + running) and returns the
42
+ * outcome.
43
+ */
44
+ export type SosInjector = (terminalId: string, text: string) => { error?: string; ok: boolean };
45
+
46
+ /** A WS listener registered against a super-session. */
47
+ export type SosListener = (msg: SosServerMessage) => void;
48
+
49
+ export interface SosMember {
50
+ /** Free-text capability tag, e.g. 'frontend' | 'backend' | '' (unused in MVP routing). */
51
+ capability: string;
52
+ /** Random hex id, primary key in sos_sessions. */
53
+ id: string;
54
+ provider: SessionSource;
55
+ registeredAt: string;
56
+ /** Watcher key: file path for file-backed providers, session id for OpenCode. */
57
+ sessionKey: string;
58
+ status: SosMemberStatus;
59
+ /** PtyManager terminal id when launched via Shooter; null for externally-observed sessions. */
60
+ terminalId: null | string;
61
+ }
62
+
63
+ // ── WebSocket protocol (/ws/super-session/:id) ──────────────────────────
64
+
65
+ /** Lifecycle status of a SoS member's underlying agent session. */
66
+ export type SosMemberStatus = 'Compacting' | 'Finished' | 'Idle' | 'Waiting' | 'Working';
67
+
68
+ /**
69
+ * A static routing rule (Phase 2). The coordinator evaluates these in ascending
70
+ * `priority` against each new member message; the first match decides the
71
+ * action. `fromMemberId` may be 'ANY'; an empty `matchPattern` matches all text.
72
+ */
73
+ export interface SosRoutingRule {
74
+ action: 'block' | 'escalate' | 'relay';
75
+ fromMemberId: string;
76
+ id: string;
77
+ matchPattern: string;
78
+ priority: number;
79
+ toMemberId: string;
80
+ }
81
+
82
+ /** A message broadcast from the coordinator to subscribed SoS clients. */
83
+ export type SosServerMessage =
84
+ | {
85
+ decision: 'approved' | 'denied' | 'expired';
86
+ relayId: string;
87
+ type: 'sos-relay-resolved';
88
+ }
89
+ | { entries: SosTranscriptEntry[]; type: 'sos-history' }
90
+ | { entry: SosTranscriptEntry; type: 'sos-message' }
91
+ | {
92
+ expiresAt: string;
93
+ fromMemberId: string;
94
+ preview: string;
95
+ relayId: string;
96
+ toMemberId: string;
97
+ type: 'sos-relay-pending';
98
+ }
99
+ | { member: SosMember; type: 'sos-member-added' }
100
+ | { memberId: string; status: SosMemberStatus; type: 'sos-member-status' }
101
+ | { memberId: string; type: 'sos-member-removed' }
102
+ | { message: string; type: 'sos-error' };
103
+
104
+ /** One message in the merged transcript, tagged with its origin member. */
105
+ export interface SosTranscriptEntry {
106
+ memberId: string;
107
+ message: ConversationMessage;
108
+ provider: SessionSource;
109
+ /** True when the coordinator injected this message (loop-guard marker). */
110
+ relayed: boolean;
111
+ }
112
+
113
+ /** A super-session: N merged agent sessions coordinated as one. */
114
+ export interface SuperSession {
115
+ createdAt: string;
116
+ id: string;
117
+ label: string;
118
+ members: SosMember[];
119
+ routingRules: SosRoutingRule[];
120
+ status: 'active' | 'archived' | 'paused';
121
+ /** Source-tagged merged transcript; in-memory ring buffer (capped). */
122
+ transcript: SosTranscriptEntry[];
123
+ }
124
+
125
+ /** Coordinator-internal runtime state for one live super-session. */
126
+ export interface SuperSessionRuntime {
127
+ /** `${fromMemberId}:${toMemberId}` -> last auto-relay epoch ms (cooldown guard). */
128
+ cooldowns: Map<string, number>;
129
+ listeners: Set<SosListener>;
130
+ /** Escalated relays awaiting human decision, keyed by relayId. */
131
+ pending: Map<string, PendingRelay>;
132
+ session: SuperSession;
133
+ unsubscribes: Map<string, () => void>;
134
+ }
@@ -10,6 +10,7 @@
10
10
  import DashboardSvg from '$lib/assets/icons/dashboard.svg?raw';
11
11
  import SettingsSvg from '$lib/assets/icons/settings.svg?raw';
12
12
  import TerminalSvg from '$lib/assets/icons/terminal.svg?raw';
13
+ import ToolSvg from '$lib/assets/icons/tool.svg?raw';
13
14
  import { Button, Icon, Pill } from '@juspay/svelte-ui-components';
14
15
  import { onMount, type Snippet } from 'svelte';
15
16
 
@@ -133,6 +134,10 @@
133
134
  <Icon svg={TerminalSvg} classes="icon-26" />
134
135
  <span>Terminals</span>
135
136
  </a>
137
+ <a href="/sos" class="tab-item" class:active={$page.url.pathname.startsWith('/sos')}>
138
+ <Icon svg={ToolSvg} classes="icon-26" />
139
+ <span>SoS</span>
140
+ </a>
136
141
  </div>
137
142
  </nav>
138
143
  </div>
@@ -191,6 +196,7 @@
191
196
  }
192
197
  .tab-item {
193
198
  display: flex;
199
+ flex: 1;
194
200
  flex-direction: column;
195
201
  align-items: center;
196
202
  justify-content: center;
@@ -199,7 +205,7 @@
199
205
  font-size: 11px;
200
206
  font-weight: 500;
201
207
  text-decoration: none;
202
- padding: 6px 36px;
208
+ padding: 6px 8px;
203
209
  border-radius: var(--radius-md);
204
210
  transition: color var(--transition-fast);
205
211
  user-select: none;
@@ -226,7 +232,8 @@
226
232
  height: 60px;
227
233
  }
228
234
  .tab-item {
229
- padding: 6px 28px;
235
+ padding: 6px 8px;
236
+ min-width: 0;
230
237
  font-size: 10px;
231
238
  gap: 3px;
232
239
  min-height: 44px;
@@ -36,8 +36,9 @@ export const POST: RequestHandler = async ({ request }) => {
36
36
  }
37
37
 
38
38
  // sessionId becomes a process argument (e.g. `codex resume <id>`); restrict it
39
- // to safe identifier characters to prevent argument/path injection.
40
- if (!/^[A-Za-z0-9_-]+$/.test(sessionId)) {
39
+ // to safe identifier chars dots included, since cursor/copilot/amp IDs use
40
+ // them — but no path separators, which prevents argument/path injection.
41
+ if (!/^[A-Za-z0-9_.-]+$/.test(sessionId)) {
41
42
  return json({ error: 'Invalid sessionId format' }, { status: 400 });
42
43
  }
43
44
 
@@ -76,9 +77,12 @@ export const POST: RequestHandler = async ({ request }) => {
76
77
  const existing = ptyManager.list().find(
77
78
  (t) =>
78
79
  t.status === 'running' &&
79
- // Claude: <id>.jsonl ; Codex rollout: rollout-<ts>-<id>.jsonl ; OpenCode: session id
80
+ // Claude / Cursor / Qwen: <id>.jsonl ; Codex rollout: rollout-<ts>-<id>.jsonl ;
81
+ // Copilot: <id>.jsonl OR <id>/events.jsonl ; Amp: T-<id>.json ; OpenCode: session id
80
82
  (t.sessionFile?.endsWith(`/${sessionId}.jsonl`) ||
81
83
  t.sessionFile?.endsWith(`-${sessionId}.jsonl`) ||
84
+ t.sessionFile?.endsWith(`/${sessionId}/events.jsonl`) ||
85
+ t.sessionFile?.endsWith(`/T-${sessionId}.json`) ||
82
86
  t.openCodeSessionId === sessionId)
83
87
  );
84
88
 
@@ -0,0 +1,36 @@
1
+ import { validateAuth } from '$lib/modules/server/auth';
2
+ import { sosCoordinator } from '$lib/modules/server/sos/coordinator';
3
+ import { json } from '@sveltejs/kit';
4
+
5
+ import type { RequestHandler } from './$types';
6
+
7
+ /** GET /api/sos — list all super-sessions. */
8
+ export const GET: RequestHandler = ({ request }) => {
9
+ const authError = validateAuth(request);
10
+ if (authError) {
11
+ return authError;
12
+ }
13
+ return json({ superSessions: sosCoordinator.listSuperSessions() });
14
+ };
15
+
16
+ /** POST /api/sos — create a new super-session. */
17
+ export const POST: RequestHandler = async ({ request }) => {
18
+ const authError = validateAuth(request);
19
+ if (authError) {
20
+ return authError;
21
+ }
22
+ let body: unknown;
23
+ try {
24
+ body = await request.json();
25
+ } catch {
26
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
27
+ }
28
+ if (typeof body !== 'object' || body === null) {
29
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
30
+ }
31
+ const payload = body as { label?: unknown };
32
+ const label =
33
+ typeof payload.label === 'string' && payload.label.trim() ? payload.label.trim() : 'Untitled';
34
+ const session = sosCoordinator.createSuperSession(label);
35
+ return json(session, { status: 201 });
36
+ };
@@ -0,0 +1,55 @@
1
+ import { validateAuth } from '$lib/modules/server/auth';
2
+ import { sosCoordinator } from '$lib/modules/server/sos/coordinator';
3
+ import { json } from '@sveltejs/kit';
4
+
5
+ import type { RequestHandler } from './$types';
6
+
7
+ /** GET /api/sos/[id] — fetch one super-session (incl. members + transcript). */
8
+ export const GET: RequestHandler = ({ params, request }) => {
9
+ const authError = validateAuth(request);
10
+ if (authError) {
11
+ return authError;
12
+ }
13
+ const session = sosCoordinator.getSuperSession(params.id ?? '');
14
+ if (!session) {
15
+ return json({ error: 'Super-session not found' }, { status: 404 });
16
+ }
17
+ return json(session);
18
+ };
19
+
20
+ /** PATCH /api/sos/[id] — update lifecycle status (active | paused | archived). */
21
+ export const PATCH: RequestHandler = async ({ params, request }) => {
22
+ const authError = validateAuth(request);
23
+ if (authError) {
24
+ return authError;
25
+ }
26
+ let body: unknown;
27
+ try {
28
+ body = await request.json();
29
+ } catch {
30
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
31
+ }
32
+ if (typeof body !== 'object' || body === null) {
33
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
34
+ }
35
+ const payload = body as { status?: unknown };
36
+ if (payload.status !== 'active' && payload.status !== 'paused' && payload.status !== 'archived') {
37
+ return json({ error: 'status must be active | paused | archived' }, { status: 400 });
38
+ }
39
+ if (!sosCoordinator.setStatus(params.id ?? '', payload.status)) {
40
+ return json({ error: 'Super-session not found' }, { status: 404 });
41
+ }
42
+ return json({ id: params.id, status: payload.status });
43
+ };
44
+
45
+ /** DELETE /api/sos/[id] — tear down a super-session (unsubscribes all members). */
46
+ export const DELETE: RequestHandler = ({ params, request }) => {
47
+ const authError = validateAuth(request);
48
+ if (authError) {
49
+ return authError;
50
+ }
51
+ if (!sosCoordinator.deleteSuperSession(params.id ?? '')) {
52
+ return json({ error: 'Super-session not found' }, { status: 404 });
53
+ }
54
+ return json({ deleted: true });
55
+ };
@@ -0,0 +1,44 @@
1
+ import { validateAuth } from '$lib/modules/server/auth';
2
+ import { sosCoordinator } from '$lib/modules/server/sos/coordinator';
3
+ import { json } from '@sveltejs/kit';
4
+
5
+ import type { RequestHandler } from './$types';
6
+
7
+ const MAX_RELAY_TEXT = 10240; // 10 KB, same cap as /ws/session send-input
8
+
9
+ /**
10
+ * POST /api/sos/[id]/inject — human-initiated relay: inject text into a member's
11
+ * Shooter-owned terminal and record it in the merged transcript.
12
+ */
13
+ export const POST: RequestHandler = async ({ params, request }) => {
14
+ const authError = validateAuth(request);
15
+ if (authError) {
16
+ return authError;
17
+ }
18
+ let body: unknown;
19
+ try {
20
+ body = await request.json();
21
+ } catch {
22
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
23
+ }
24
+ if (typeof body !== 'object' || body === null) {
25
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
26
+ }
27
+ const payload = body as { text?: unknown; toMemberId?: unknown };
28
+ if (typeof payload.toMemberId !== 'string' || payload.toMemberId.length === 0) {
29
+ return json({ error: 'toMemberId is required (string)' }, { status: 400 });
30
+ }
31
+ if (typeof payload.text !== 'string' || payload.text.length === 0) {
32
+ return json({ error: 'text is required (string)' }, { status: 400 });
33
+ }
34
+ if (Buffer.byteLength(payload.text, 'utf8') > MAX_RELAY_TEXT) {
35
+ return json({ error: `text exceeds ${MAX_RELAY_TEXT} bytes` }, { status: 413 });
36
+ }
37
+
38
+ const error = sosCoordinator.relayForward(params.id ?? '', payload.toMemberId, payload.text);
39
+ if (error) {
40
+ const status = error.includes('not found') ? 404 : 400;
41
+ return json({ error }, { status });
42
+ }
43
+ return json({ ok: true });
44
+ };
@@ -0,0 +1,47 @@
1
+ import { validateAuth } from '$lib/modules/server/auth';
2
+ import { sosCoordinator } from '$lib/modules/server/sos/coordinator';
3
+ import { isValidProvider, isValidSessionKey } from '$lib/modules/server/ws/super-session-handler';
4
+ import { json } from '@sveltejs/kit';
5
+
6
+ import type { RequestHandler } from './$types';
7
+
8
+ /** POST /api/sos/[id]/members — add a member session to a super-session. */
9
+ export const POST: RequestHandler = async ({ params, request }) => {
10
+ const authError = validateAuth(request);
11
+ if (authError) {
12
+ return authError;
13
+ }
14
+ let body: unknown;
15
+ try {
16
+ body = await request.json();
17
+ } catch {
18
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
19
+ }
20
+ if (typeof body !== 'object' || body === null) {
21
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
22
+ }
23
+ const payload = body as {
24
+ capability?: unknown;
25
+ provider?: unknown;
26
+ sessionKey?: unknown;
27
+ terminalId?: unknown;
28
+ };
29
+
30
+ if (typeof payload.sessionKey !== 'string' || !isValidSessionKey(payload.sessionKey)) {
31
+ return json({ error: 'sessionKey must be a session id or a path under home' }, { status: 400 });
32
+ }
33
+ if (typeof payload.provider !== 'string' || !isValidProvider(payload.provider)) {
34
+ return json({ error: 'provider must be a known session source' }, { status: 400 });
35
+ }
36
+
37
+ const member = sosCoordinator.addMember(params.id ?? '', {
38
+ capability: typeof payload.capability === 'string' ? payload.capability : undefined,
39
+ provider: payload.provider,
40
+ sessionKey: payload.sessionKey,
41
+ terminalId: typeof payload.terminalId === 'string' ? payload.terminalId : null,
42
+ });
43
+ if (!member) {
44
+ return json({ error: 'Super-session not found' }, { status: 404 });
45
+ }
46
+ return json(member, { status: 201 });
47
+ };
@@ -0,0 +1,17 @@
1
+ import { validateAuth } from '$lib/modules/server/auth';
2
+ import { sosCoordinator } from '$lib/modules/server/sos/coordinator';
3
+ import { json } from '@sveltejs/kit';
4
+
5
+ import type { RequestHandler } from './$types';
6
+
7
+ /** DELETE /api/sos/[id]/members/[mid] — remove a member (unsubscribes its watcher). */
8
+ export const DELETE: RequestHandler = ({ params, request }) => {
9
+ const authError = validateAuth(request);
10
+ if (authError) {
11
+ return authError;
12
+ }
13
+ if (!sosCoordinator.removeMember(params.id ?? '', params.mid ?? '')) {
14
+ return json({ error: 'Super-session or member not found' }, { status: 404 });
15
+ }
16
+ return json({ removed: true });
17
+ };
@@ -0,0 +1,85 @@
1
+ import type { SosRoutingRule } from '$lib/types';
2
+
3
+ import { validateAuth } from '$lib/modules/server/auth';
4
+ import { sosCoordinator } from '$lib/modules/server/sos/coordinator';
5
+ import { json } from '@sveltejs/kit';
6
+ import { randomBytes } from 'crypto';
7
+
8
+ import type { RequestHandler } from './$types';
9
+
10
+ const ACTIONS = new Set(['block', 'escalate', 'relay']);
11
+
12
+ /** GET /api/sos/[id]/rules — current routing rules. */
13
+ export const GET: RequestHandler = ({ params, request }) => {
14
+ const authError = validateAuth(request);
15
+ if (authError) {
16
+ return authError;
17
+ }
18
+ const rules = sosCoordinator.getRoutingRules(params.id ?? '');
19
+ if (rules === null) {
20
+ return json({ error: 'Super-session not found' }, { status: 404 });
21
+ }
22
+ return json({ routingRules: rules });
23
+ };
24
+
25
+ /** PATCH /api/sos/[id]/rules — replace the routing rules. */
26
+ export const PATCH: RequestHandler = async ({ params, request }) => {
27
+ const authError = validateAuth(request);
28
+ if (authError) {
29
+ return authError;
30
+ }
31
+ let body: unknown;
32
+ try {
33
+ body = await request.json();
34
+ } catch {
35
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
36
+ }
37
+ if (typeof body !== 'object' || body === null) {
38
+ return json({ error: 'Invalid JSON in request body' }, { status: 400 });
39
+ }
40
+ const payload = body as { routingRules?: unknown };
41
+ if (!Array.isArray(payload.routingRules)) {
42
+ return json({ error: 'routingRules must be an array' }, { status: 400 });
43
+ }
44
+
45
+ const rules: SosRoutingRule[] = [];
46
+ for (const raw of payload.routingRules) {
47
+ const rule = normalizeRule(raw);
48
+ if (!rule) {
49
+ return json(
50
+ { error: 'Each rule needs fromMemberId, toMemberId, and action' },
51
+ { status: 400 }
52
+ );
53
+ }
54
+ rules.push(rule);
55
+ }
56
+
57
+ const error = sosCoordinator.setRoutingRules(params.id ?? '', rules);
58
+ if (error) {
59
+ return json({ error }, { status: error.includes('not found') ? 404 : 400 });
60
+ }
61
+ return json({ routingRules: rules });
62
+ };
63
+
64
+ function normalizeRule(raw: unknown): null | SosRoutingRule {
65
+ if (typeof raw !== 'object' || raw === null) {
66
+ return null;
67
+ }
68
+ const r = raw as Record<string, unknown>;
69
+ if (
70
+ typeof r.fromMemberId !== 'string' ||
71
+ typeof r.toMemberId !== 'string' ||
72
+ typeof r.action !== 'string' ||
73
+ !ACTIONS.has(r.action)
74
+ ) {
75
+ return null;
76
+ }
77
+ return {
78
+ action: r.action as SosRoutingRule['action'],
79
+ fromMemberId: r.fromMemberId,
80
+ id: typeof r.id === 'string' && r.id ? r.id : randomBytes(6).toString('hex'),
81
+ matchPattern: typeof r.matchPattern === 'string' ? r.matchPattern : '',
82
+ priority: typeof r.priority === 'number' ? r.priority : 100,
83
+ toMemberId: r.toMemberId,
84
+ };
85
+ }
@@ -0,0 +1,195 @@
1
+ <script lang="ts">
2
+ import type { ShooterConfig, SuperSession } from '$lib/types';
3
+
4
+ import { goto } from '$app/navigation';
5
+ import { Banner, Button, EmptyState, Input, Pill } from '@juspay/svelte-ui-components';
6
+ import { onMount } from 'svelte';
7
+
8
+ let sessions = $state<SuperSession[]>([]);
9
+ let loading = $state(true);
10
+ let error = $state('');
11
+ let newLabel = $state('');
12
+ let creating = $state(false);
13
+
14
+ function getConfig(): null | ShooterConfig {
15
+ try {
16
+ const saved = localStorage.getItem('shooter_config');
17
+ return saved ? (JSON.parse(saved) as ShooterConfig) : null;
18
+ } catch {
19
+ return null;
20
+ }
21
+ }
22
+
23
+ async function loadSessions(): Promise<void> {
24
+ const config = getConfig();
25
+ if (!config) {
26
+ error = 'No configuration found. Open Settings first.';
27
+ loading = false;
28
+ return;
29
+ }
30
+ try {
31
+ const res = await fetch('/api/sos', {
32
+ headers: { Authorization: `Bearer ${config.apiKey}` },
33
+ });
34
+ if (!res.ok) {
35
+ error = `Failed to load (${res.status})`;
36
+ return;
37
+ }
38
+ const data = (await res.json()) as { superSessions: SuperSession[] };
39
+ sessions = data.superSessions ?? [];
40
+ } catch {
41
+ error = 'Network error — is the server running?';
42
+ } finally {
43
+ loading = false;
44
+ }
45
+ }
46
+
47
+ async function create(): Promise<void> {
48
+ const config = getConfig();
49
+ if (!config || !newLabel.trim()) {
50
+ return;
51
+ }
52
+ creating = true;
53
+ try {
54
+ const res = await fetch('/api/sos', {
55
+ body: JSON.stringify({ label: newLabel.trim() }),
56
+ headers: { Authorization: `Bearer ${config.apiKey}`, 'Content-Type': 'application/json' },
57
+ method: 'POST',
58
+ });
59
+ if (res.ok) {
60
+ const created = (await res.json()) as SuperSession;
61
+ void goto(`/sos/${created.id}`);
62
+ } else {
63
+ const d = (await res.json().catch(() => ({}))) as { error?: string };
64
+ error = d.error ?? `Failed to create (${res.status})`;
65
+ }
66
+ } catch {
67
+ error = 'Network error — could not create super-session';
68
+ } finally {
69
+ creating = false;
70
+ }
71
+ }
72
+
73
+ onMount(() => {
74
+ void loadSessions();
75
+ });
76
+ </script>
77
+
78
+ <svelte:head><title>Session Over Sessions - Shooter</title></svelte:head>
79
+
80
+ <main class="main sos-list">
81
+ <div class="page-head">
82
+ <div>
83
+ <h1>Session Over Sessions</h1>
84
+ <p class="subtitle">Coordinate multiple running agents as one super-session.</p>
85
+ </div>
86
+ </div>
87
+
88
+ <div class="create-row">
89
+ <Input
90
+ bind:value={newLabel}
91
+ dataType="text"
92
+ placeholder="New super-session label…"
93
+ classes="sos-create-input"
94
+ />
95
+ <Button
96
+ text={creating ? 'Creating…' : 'Create'}
97
+ disabled={creating || !newLabel.trim()}
98
+ onclick={create}
99
+ classes="btn-create"
100
+ />
101
+ </div>
102
+
103
+ {#if error}
104
+ <Banner text={error} classes="banner-error" />
105
+ {/if}
106
+
107
+ {#if loading}
108
+ <p class="muted">Loading…</p>
109
+ {:else if sessions.length === 0}
110
+ <EmptyState
111
+ title="No super-sessions yet"
112
+ description="Create one above, then add running agent sessions as members."
113
+ />
114
+ {:else}
115
+ <div class="ss-grid">
116
+ {#each sessions as ss (ss.id)}
117
+ <a class="ss-card" href={`/sos/${ss.id}`}>
118
+ <div class="ss-card-head">
119
+ <span class="ss-label">{ss.label}</span>
120
+ <Pill text={ss.status} classes="pill-status-unknown" />
121
+ </div>
122
+ <div class="ss-meta">
123
+ <span>{ss.members.length} member{ss.members.length === 1 ? '' : 's'}</span>
124
+ <span>{ss.routingRules.length} rule{ss.routingRules.length === 1 ? '' : 's'}</span>
125
+ </div>
126
+ </a>
127
+ {/each}
128
+ </div>
129
+ {/if}
130
+ </main>
131
+
132
+ <style>
133
+ .sos-list {
134
+ max-width: 720px;
135
+ margin: 0 auto;
136
+ padding: var(--space-5) var(--space-4);
137
+ }
138
+ .page-head h1 {
139
+ font-size: var(--text-2xl);
140
+ font-weight: 600;
141
+ color: var(--text-primary);
142
+ margin: 0;
143
+ }
144
+ .subtitle {
145
+ color: var(--text-secondary);
146
+ font-size: var(--text-sm);
147
+ margin: var(--space-1) 0 var(--space-4);
148
+ }
149
+ .create-row {
150
+ display: flex;
151
+ gap: var(--space-2);
152
+ align-items: center;
153
+ margin-bottom: var(--space-4);
154
+ }
155
+ :global(.sos-create-input) {
156
+ flex: 1;
157
+ --input-container-margin: 0;
158
+ }
159
+ .muted {
160
+ color: var(--text-tertiary);
161
+ }
162
+ .ss-grid {
163
+ display: grid;
164
+ gap: var(--space-3);
165
+ }
166
+ .ss-card {
167
+ display: block;
168
+ padding: var(--space-4);
169
+ background: var(--component-bg);
170
+ border: 1px solid var(--border);
171
+ border-radius: var(--radius-lg);
172
+ text-decoration: none;
173
+ transition: border-color var(--transition-fast);
174
+ }
175
+ .ss-card:hover {
176
+ border-color: var(--border-hover);
177
+ }
178
+ .ss-card-head {
179
+ display: flex;
180
+ align-items: center;
181
+ justify-content: space-between;
182
+ gap: var(--space-2);
183
+ }
184
+ .ss-label {
185
+ font-weight: 600;
186
+ color: var(--text-primary);
187
+ }
188
+ .ss-meta {
189
+ display: flex;
190
+ gap: var(--space-3);
191
+ color: var(--text-tertiary);
192
+ font-size: var(--text-xs);
193
+ margin-top: var(--space-2);
194
+ }
195
+ </style>