@juspay/shooter 1.18.0 → 1.20.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 (367) hide show
  1. package/build/client/_app/immutable/assets/{0.NV8k8wxG.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.NV8k8wxG.css.gz → 0.BwNtE8TX.css.gz} +0 -0
  4. package/build/client/_app/immutable/assets/11.F10lvwyh.css +1 -0
  5. package/build/client/_app/immutable/assets/11.F10lvwyh.css.br +0 -0
  6. package/build/client/_app/immutable/assets/11.F10lvwyh.css.gz +0 -0
  7. package/build/client/_app/immutable/assets/8.BYgAX7hR.css +1 -0
  8. package/build/client/_app/immutable/assets/8.BYgAX7hR.css.br +0 -0
  9. package/build/client/_app/immutable/assets/8.BYgAX7hR.css.gz +0 -0
  10. package/build/client/_app/immutable/assets/9.DV6pZunn.css +1 -0
  11. package/build/client/_app/immutable/assets/9.DV6pZunn.css.br +0 -0
  12. package/build/client/_app/immutable/assets/9.DV6pZunn.css.gz +0 -0
  13. package/build/client/_app/immutable/chunks/{DZQMsHM5.js → 2rBV5OkJ.js} +1 -1
  14. package/build/client/_app/immutable/chunks/2rBV5OkJ.js.br +0 -0
  15. package/build/client/_app/immutable/chunks/2rBV5OkJ.js.gz +0 -0
  16. package/build/client/_app/immutable/chunks/{B9WQy_3X.js → BB2l8o4X.js} +1 -1
  17. package/build/client/_app/immutable/chunks/BB2l8o4X.js.br +0 -0
  18. package/build/client/_app/immutable/chunks/BB2l8o4X.js.gz +0 -0
  19. package/build/client/_app/immutable/chunks/{Cg3dlX05.js → BPDiEZo0.js} +2 -2
  20. package/build/client/_app/immutable/chunks/BPDiEZo0.js.br +0 -0
  21. package/build/client/_app/immutable/chunks/BPDiEZo0.js.gz +0 -0
  22. package/build/client/_app/immutable/chunks/{C_9BZILB.js → BcpydfqI.js} +1 -1
  23. package/build/client/_app/immutable/chunks/BcpydfqI.js.br +0 -0
  24. package/build/client/_app/immutable/chunks/BcpydfqI.js.gz +0 -0
  25. package/build/client/_app/immutable/chunks/{BRqaaL5D.js → BvmdJful.js} +1 -1
  26. package/build/client/_app/immutable/chunks/BvmdJful.js.br +0 -0
  27. package/build/client/_app/immutable/chunks/BvmdJful.js.gz +0 -0
  28. package/build/client/_app/immutable/chunks/C_YNQL8b.js +3 -0
  29. package/build/client/_app/immutable/chunks/C_YNQL8b.js.br +0 -0
  30. package/build/client/_app/immutable/chunks/C_YNQL8b.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/{8lO1IL7u.js → DIZ3Qst5.js} +1 -1
  35. package/build/client/_app/immutable/chunks/DIZ3Qst5.js.br +0 -0
  36. package/build/client/_app/immutable/chunks/{8lO1IL7u.js.gz → DIZ3Qst5.js.gz} +0 -0
  37. package/build/client/_app/immutable/chunks/{DJvX78LW.js → DT4H19pV.js} +1 -1
  38. package/build/client/_app/immutable/chunks/DT4H19pV.js.br +0 -0
  39. package/build/client/_app/immutable/chunks/DT4H19pV.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/J5-Cr5oR.js +6 -0
  44. package/build/client/_app/immutable/chunks/J5-Cr5oR.js.br +0 -0
  45. package/build/client/_app/immutable/chunks/J5-Cr5oR.js.gz +0 -0
  46. package/build/client/_app/immutable/chunks/{DZvnhU_8.js → ZS5XYDx_.js} +2 -2
  47. package/build/client/_app/immutable/chunks/ZS5XYDx_.js.br +0 -0
  48. package/build/client/_app/immutable/chunks/ZS5XYDx_.js.gz +0 -0
  49. package/build/client/_app/immutable/entry/app.Bd-DfeJi.js +2 -0
  50. package/build/client/_app/immutable/entry/app.Bd-DfeJi.js.br +0 -0
  51. package/build/client/_app/immutable/entry/app.Bd-DfeJi.js.gz +0 -0
  52. package/build/client/_app/immutable/entry/start.evvp4tX7.js +1 -0
  53. package/build/client/_app/immutable/entry/start.evvp4tX7.js.br +2 -0
  54. package/build/client/_app/immutable/entry/start.evvp4tX7.js.gz +0 -0
  55. package/build/client/_app/immutable/nodes/0.Bl-1LQWM.js +10 -0
  56. package/build/client/_app/immutable/nodes/0.Bl-1LQWM.js.br +0 -0
  57. package/build/client/_app/immutable/nodes/0.Bl-1LQWM.js.gz +0 -0
  58. package/build/client/_app/immutable/nodes/{1.C4eFlqSB.js → 1.DT4dq6Ay.js} +1 -1
  59. package/build/client/_app/immutable/nodes/1.DT4dq6Ay.js.br +0 -0
  60. package/build/client/_app/immutable/nodes/1.DT4dq6Ay.js.gz +0 -0
  61. package/build/client/_app/immutable/nodes/{8.Bs362gyb.js → 10.CF7RGXpe.js} +2 -2
  62. package/build/client/_app/immutable/nodes/10.CF7RGXpe.js.br +0 -0
  63. package/build/client/_app/immutable/nodes/10.CF7RGXpe.js.gz +0 -0
  64. package/build/client/_app/immutable/nodes/11.BV_G7yLI.js +2 -0
  65. package/build/client/_app/immutable/nodes/11.BV_G7yLI.js.br +0 -0
  66. package/build/client/_app/immutable/nodes/11.BV_G7yLI.js.gz +0 -0
  67. package/build/client/_app/immutable/nodes/{2.CdC092Za.js → 2.DcRhsjYp.js} +2 -2
  68. package/build/client/_app/immutable/nodes/2.DcRhsjYp.js.br +0 -0
  69. package/build/client/_app/immutable/nodes/2.DcRhsjYp.js.gz +0 -0
  70. package/build/client/_app/immutable/nodes/{3.Dhf4ZWW0.js → 3.0MMe3oxR.js} +3 -3
  71. package/build/client/_app/immutable/nodes/3.0MMe3oxR.js.br +0 -0
  72. package/build/client/_app/immutable/nodes/3.0MMe3oxR.js.gz +0 -0
  73. package/build/client/_app/immutable/nodes/{4.BSVqdrrD.js → 4.CBX9A3ka.js} +2 -2
  74. package/build/client/_app/immutable/nodes/4.CBX9A3ka.js.br +0 -0
  75. package/build/client/_app/immutable/nodes/4.CBX9A3ka.js.gz +0 -0
  76. package/build/client/_app/immutable/nodes/{5.Cfj35gpY.js → 5.DIVKuZc9.js} +1 -1
  77. package/build/client/_app/immutable/nodes/5.DIVKuZc9.js.br +0 -0
  78. package/build/client/_app/immutable/nodes/5.DIVKuZc9.js.gz +0 -0
  79. package/build/client/_app/immutable/nodes/{6.B3SEB_li.js → 6.ComiWlV6.js} +1 -1
  80. package/build/client/_app/immutable/nodes/6.ComiWlV6.js.br +0 -0
  81. package/build/client/_app/immutable/nodes/6.ComiWlV6.js.gz +0 -0
  82. package/build/client/_app/immutable/nodes/{7.DV8cJ1lX.js → 7.vkPx1kVP.js} +1 -1
  83. package/build/client/_app/immutable/nodes/7.vkPx1kVP.js.br +0 -0
  84. package/build/client/_app/immutable/nodes/7.vkPx1kVP.js.gz +0 -0
  85. package/build/client/_app/immutable/nodes/8.Bmr3sWbS.js +1 -0
  86. package/build/client/_app/immutable/nodes/8.Bmr3sWbS.js.br +0 -0
  87. package/build/client/_app/immutable/nodes/8.Bmr3sWbS.js.gz +0 -0
  88. package/build/client/_app/immutable/nodes/9.CAJucyeI.js +2 -0
  89. package/build/client/_app/immutable/nodes/9.CAJucyeI.js.br +0 -0
  90. package/build/client/_app/immutable/nodes/9.CAJucyeI.js.gz +0 -0
  91. package/build/client/_app/version.json +1 -1
  92. package/build/client/_app/version.json.br +0 -0
  93. package/build/client/_app/version.json.gz +0 -0
  94. package/build/server/chunks/{0-Cd7jY0a7.js → 0-DDGB6CRT.js} +4 -4
  95. package/build/server/chunks/{0-Cd7jY0a7.js.map → 0-DDGB6CRT.js.map} +1 -1
  96. package/build/server/chunks/1-DEjonQXD.js +9 -0
  97. package/build/server/chunks/{1-C4BOGoJY.js.map → 1-DEjonQXD.js.map} +1 -1
  98. package/build/server/chunks/10-BK1kiiiw.js +9 -0
  99. package/build/server/chunks/10-BK1kiiiw.js.map +1 -0
  100. package/build/server/chunks/11-CJPjkEF3.js +9 -0
  101. package/build/server/chunks/11-CJPjkEF3.js.map +1 -0
  102. package/build/server/chunks/{2-Ba0mNwJ6.js → 2-RLnhlWh5.js} +3 -3
  103. package/build/server/chunks/{2-Ba0mNwJ6.js.map → 2-RLnhlWh5.js.map} +1 -1
  104. package/build/server/chunks/{3-Pg8t1uJU.js → 3-Dd4pJBqZ.js} +3 -3
  105. package/build/server/chunks/{3-Pg8t1uJU.js.map → 3-Dd4pJBqZ.js.map} +1 -1
  106. package/build/server/chunks/{4-BtYdKCVW.js → 4-Bb5VFhsO.js} +3 -3
  107. package/build/server/chunks/{4-BtYdKCVW.js.map → 4-Bb5VFhsO.js.map} +1 -1
  108. package/build/server/chunks/{5-CvJK3PiH.js → 5-oNoWuIsn.js} +3 -3
  109. package/build/server/chunks/{5-CvJK3PiH.js.map → 5-oNoWuIsn.js.map} +1 -1
  110. package/build/server/chunks/6-DdRMnKNa.js +9 -0
  111. package/build/server/chunks/{6-D8xbnTSo.js.map → 6-DdRMnKNa.js.map} +1 -1
  112. package/build/server/chunks/7-vLOMMetm.js +9 -0
  113. package/build/server/chunks/{7-CkVK06S0.js.map → 7-vLOMMetm.js.map} +1 -1
  114. package/build/server/chunks/8-rJyiQLFs.js +9 -0
  115. package/build/server/chunks/8-rJyiQLFs.js.map +1 -0
  116. package/build/server/chunks/9-CVSNNYED.js +9 -0
  117. package/build/server/chunks/9-CVSNNYED.js.map +1 -0
  118. package/build/server/chunks/Banner-BgaAs1rs.js +90 -0
  119. package/build/server/chunks/Banner-BgaAs1rs.js.map +1 -0
  120. package/build/server/chunks/{Button-B5dU-ntz.js → Button-D0hZ7JYt.js} +2 -2
  121. package/build/server/chunks/Button-D0hZ7JYt.js.map +1 -0
  122. package/build/server/chunks/{Icon-C7Ml3GX6.js → Icon-D0GBnDcs.js} +3 -3
  123. package/build/server/chunks/Icon-D0GBnDcs.js.map +1 -0
  124. package/build/server/chunks/{Input-CPGO0sbS.js → Input-OmIiydSx.js} +2 -2
  125. package/build/server/chunks/Input-OmIiydSx.js.map +1 -0
  126. package/build/server/chunks/{Pill-CcrtCejm.js → Pill-4xJ-VhAA.js} +3 -3
  127. package/build/server/chunks/Pill-4xJ-VhAA.js.map +1 -0
  128. package/build/server/chunks/{Shimmer-C5jkvGr1.js → Shimmer-Dw2uvTC1.js} +2 -2
  129. package/build/server/chunks/Shimmer-Dw2uvTC1.js.map +1 -0
  130. package/build/server/chunks/{_error.svelte-CSIxs-ab.js → _error.svelte-CZnkxeLr.js} +8 -8
  131. package/build/server/chunks/_error.svelte-CZnkxeLr.js.map +1 -0
  132. package/build/server/chunks/{_layout.svelte-noB4j-v2.js → _layout.svelte-DfgNGGiM.js} +16 -11
  133. package/build/server/chunks/_layout.svelte-DfgNGGiM.js.map +1 -0
  134. package/build/server/chunks/{_page.svelte-DnTpPnPR.js → _page.svelte-BLo2v_8E.js} +7 -88
  135. package/build/server/chunks/_page.svelte-BLo2v_8E.js.map +1 -0
  136. package/build/server/chunks/_page.svelte-BTlfUsBp.js +43 -0
  137. package/build/server/chunks/_page.svelte-BTlfUsBp.js.map +1 -0
  138. package/build/server/chunks/{_page.svelte-BV0XyYJZ.js → _page.svelte-BX2FMgSg.js} +4 -4
  139. package/build/server/chunks/_page.svelte-BX2FMgSg.js.map +1 -0
  140. package/build/server/chunks/{_page.svelte-BUkm2304.js → _page.svelte-C7B0qdrC.js} +5 -5
  141. package/build/server/chunks/_page.svelte-C7B0qdrC.js.map +1 -0
  142. package/build/server/chunks/{_page.svelte-Dmg-RFCg.js → _page.svelte-CE7COWnF.js} +7 -7
  143. package/build/server/chunks/_page.svelte-CE7COWnF.js.map +1 -0
  144. package/build/server/chunks/{_page.svelte-BfB8maoc.js → _page.svelte-CWsjjd4l.js} +9 -9
  145. package/build/server/chunks/_page.svelte-CWsjjd4l.js.map +1 -0
  146. package/build/server/chunks/_page.svelte-D5S2hkBk.js +104 -0
  147. package/build/server/chunks/_page.svelte-D5S2hkBk.js.map +1 -0
  148. package/build/server/chunks/{_page.svelte-B6qyh-K-.js → _page.svelte-D_Ey8QRG.js} +11 -11
  149. package/build/server/chunks/_page.svelte-D_Ey8QRG.js.map +1 -0
  150. package/build/server/chunks/{_page.svelte-C60lAagP.js → _page.svelte-dabsQl9c.js} +210 -9
  151. package/build/server/chunks/_page.svelte-dabsQl9c.js.map +1 -0
  152. package/build/server/chunks/{_page.svelte-DuzZr5dA.js → _page.svelte-tBuIq8Pg.js} +11 -11
  153. package/build/server/chunks/_page.svelte-tBuIq8Pg.js.map +1 -0
  154. package/build/server/chunks/{_server.ts-CyjDrcZN.js → _server.ts-AnBXfZXh.js} +10 -2
  155. package/build/server/chunks/_server.ts-AnBXfZXh.js.map +1 -0
  156. package/build/server/chunks/_server.ts-B-evHL2q.js +13 -0
  157. package/build/server/chunks/_server.ts-B-evHL2q.js.map +1 -0
  158. package/build/server/chunks/_server.ts-B2wIgsW4.js +95 -0
  159. package/build/server/chunks/_server.ts-B2wIgsW4.js.map +1 -0
  160. package/build/server/chunks/_server.ts-BaaY7Z9D.js +77 -0
  161. package/build/server/chunks/_server.ts-BaaY7Z9D.js.map +1 -0
  162. package/build/server/chunks/_server.ts-C0317RBD.js +57 -0
  163. package/build/server/chunks/_server.ts-C0317RBD.js.map +1 -0
  164. package/build/server/chunks/{_server.ts-Bu3s5hfv.js → _server.ts-CJGyN8mw.js} +18 -10
  165. package/build/server/chunks/_server.ts-CJGyN8mw.js.map +1 -0
  166. package/build/server/chunks/_server.ts-CVPZOpiv.js +23 -0
  167. package/build/server/chunks/_server.ts-CVPZOpiv.js.map +1 -0
  168. package/build/server/chunks/{_server.ts-DZgfQKiH.js → _server.ts-D9ir7u24.js} +2 -2
  169. package/build/server/chunks/{_server.ts-DZgfQKiH.js.map → _server.ts-D9ir7u24.js.map} +1 -1
  170. package/build/server/chunks/{_server.ts-DZP2lhaY.js → _server.ts-DEx9-epI.js} +20 -8
  171. package/build/server/chunks/_server.ts-DEx9-epI.js.map +1 -0
  172. package/build/server/chunks/{_server.ts-BA_uWcPw.js → _server.ts-DKNIsQeH.js} +6 -4
  173. package/build/server/chunks/_server.ts-DKNIsQeH.js.map +1 -0
  174. package/build/server/chunks/_server.ts-DkZX_O9a.js +39 -0
  175. package/build/server/chunks/_server.ts-DkZX_O9a.js.map +1 -0
  176. package/build/server/chunks/_server.ts-DpRr0Tfh.js +68 -0
  177. package/build/server/chunks/_server.ts-DpRr0Tfh.js.map +1 -0
  178. package/build/server/chunks/{_server.ts-CwAjt91u.js → _server.ts-Dz9Jd9Jh.js} +6 -4
  179. package/build/server/chunks/{_server.ts-CwAjt91u.js.map → _server.ts-Dz9Jd9Jh.js.map} +1 -1
  180. package/build/server/chunks/_server.ts-Mttr0-Sl.js +48 -0
  181. package/build/server/chunks/_server.ts-Mttr0-Sl.js.map +1 -0
  182. package/build/server/chunks/{_server.ts-Bjbr7glm.js → _server.ts-QN-Bo5ql.js} +12 -5
  183. package/build/server/chunks/_server.ts-QN-Bo5ql.js.map +1 -0
  184. package/build/server/chunks/{_server.ts-BrqaMMAa.js → _server.ts-W6i3EnGX.js} +29 -6
  185. package/build/server/chunks/_server.ts-W6i3EnGX.js.map +1 -0
  186. package/build/server/chunks/{_server.ts-DZ5naqSL.js → _server.ts-bk_EeAdY.js} +6 -2
  187. package/build/server/chunks/_server.ts-bk_EeAdY.js.map +1 -0
  188. package/build/server/chunks/_server.ts-jtqWDWcf.js +45 -0
  189. package/build/server/chunks/_server.ts-jtqWDWcf.js.map +1 -0
  190. package/build/server/chunks/{cache-Me3zUAaD.js → cache-BlMaDsHi.js} +2 -2
  191. package/build/server/chunks/cache-BlMaDsHi.js.map +1 -0
  192. package/build/server/chunks/{client-CfNnl32g.js → client-Ds1brw-8.js} +4 -4
  193. package/build/server/chunks/{client-CfNnl32g.js.map → client-Ds1brw-8.js.map} +1 -1
  194. package/build/server/chunks/client2-DngLdcUc.js +7 -0
  195. package/build/server/chunks/{client2-DDP30_vY.js.map → client2-DngLdcUc.js.map} +1 -1
  196. package/build/server/chunks/coordinator-DMU_ADXf.js +530 -0
  197. package/build/server/chunks/coordinator-DMU_ADXf.js.map +1 -0
  198. package/build/server/chunks/guest-registry-t0-7Zv5q.js +39 -0
  199. package/build/server/chunks/guest-registry-t0-7Zv5q.js.map +1 -0
  200. package/build/server/chunks/{index-CJrGuxuM.js → index-CoYB03g7.js} +2 -2
  201. package/build/server/chunks/{index-CJrGuxuM.js.map → index-CoYB03g7.js.map} +1 -1
  202. package/build/server/chunks/{index-server--49oHtA0.js → index-server-Bq3cnK69.js} +2 -2
  203. package/build/server/chunks/{index-server--49oHtA0.js.map → index-server-Bq3cnK69.js.map} +1 -1
  204. package/build/server/chunks/{index2-MY7PXeAc.js → index2-dSGQ9Eaa.js} +2 -2
  205. package/build/server/chunks/{index2-MY7PXeAc.js.map → index2-dSGQ9Eaa.js.map} +1 -1
  206. package/build/server/chunks/{pty-manager-DmNSCKAr.js → pty-manager-CkZNoW1t.js} +7 -2
  207. package/build/server/chunks/pty-manager-CkZNoW1t.js.map +1 -0
  208. package/build/server/chunks/qwen-reader-DGfUbKaJ.js.map +1 -1
  209. package/build/server/chunks/{registry-Kcw2UCMv.js → registry-D4J_CuzW.js} +2 -2
  210. package/build/server/chunks/registry-D4J_CuzW.js.map +1 -0
  211. package/build/server/chunks/{root-xvQIR1Bt.js → root-D4IoFC8F.js} +2 -2
  212. package/build/server/chunks/root-D4IoFC8F.js.map +1 -0
  213. package/build/server/chunks/share-auth-BS7JuiHf.js +27 -0
  214. package/build/server/chunks/share-auth-BS7JuiHf.js.map +1 -0
  215. package/build/server/chunks/share-store-B9jMpVg0.js +127 -0
  216. package/build/server/chunks/share-store-B9jMpVg0.js.map +1 -0
  217. package/build/server/chunks/{state.svelte-RCtlkrNH.js → state.svelte-CmHqngc_.js} +3 -3
  218. package/build/server/chunks/{state.svelte-RCtlkrNH.js.map → state.svelte-CmHqngc_.js.map} +1 -1
  219. package/build/server/chunks/{stores-C-LqoonT.js → stores-CRYxfF0o.js} +4 -4
  220. package/build/server/chunks/stores-CRYxfF0o.js.map +1 -0
  221. package/build/server/chunks/super-session-handler-DPyxFgmz.js +22 -0
  222. package/build/server/chunks/super-session-handler-DPyxFgmz.js.map +1 -0
  223. package/build/server/index.js +4 -4
  224. package/build/server/index.js.map +1 -1
  225. package/build/server/manifest.js +101 -22
  226. package/build/server/manifest.js.map +1 -1
  227. package/package.json +2 -2
  228. package/scripts/e2e-all-features.sh +41 -2
  229. package/server.ts +33 -3
  230. package/src/lib/modules/client/terminal/ShareGate.svelte +96 -0
  231. package/src/lib/modules/client/terminal/ShareSheet.svelte +395 -0
  232. package/src/lib/modules/client/terminal/xterm-wrapper.ts +19 -2
  233. package/src/lib/modules/server/sos/coordinator.ts +492 -0
  234. package/src/lib/modules/server/sos/policy-gate.ts +56 -0
  235. package/src/lib/modules/server/sos/relay-store.ts +159 -0
  236. package/src/lib/modules/server/terminal/pty-input.ts +37 -0
  237. package/src/lib/modules/server/terminal/pty-manager.ts +6 -0
  238. package/src/lib/modules/server/terminal/share-auth.ts +37 -0
  239. package/src/lib/modules/server/terminal/share-store.ts +172 -0
  240. package/src/lib/modules/server/ws/guest-registry.ts +49 -0
  241. package/src/lib/modules/server/ws/server.ts +28 -4
  242. package/src/lib/modules/server/ws/session-handler.ts +24 -5
  243. package/src/lib/modules/server/ws/super-session-handler.ts +200 -0
  244. package/src/lib/modules/server/ws/terminal-handler.ts +21 -2
  245. package/src/lib/modules/server/ws/ticket-store.ts +18 -10
  246. package/src/lib/types/generated/Client.ts +25 -1
  247. package/src/lib/types/generated/Share.ts +404 -0
  248. package/src/lib/types/generated/WsProtocol.ts +73 -2
  249. package/src/lib/types/generated/index.ts +1 -0
  250. package/src/lib/types/index.ts +2 -1
  251. package/src/lib/types/sos.ts +134 -0
  252. package/src/lib/types/terminal-client.ts +19 -2
  253. package/src/lib/types/ws.ts +1 -0
  254. package/src/routes/+layout.svelte +9 -2
  255. package/src/routes/api/sos/+server.ts +36 -0
  256. package/src/routes/api/sos/[id]/+server.ts +55 -0
  257. package/src/routes/api/sos/[id]/inject/+server.ts +44 -0
  258. package/src/routes/api/sos/[id]/members/+server.ts +47 -0
  259. package/src/routes/api/sos/[id]/members/[mid]/+server.ts +17 -0
  260. package/src/routes/api/sos/[id]/rules/+server.ts +85 -0
  261. package/src/routes/api/terminals/[id]/+server.ts +14 -3
  262. package/src/routes/api/terminals/[id]/paste-image/+server.ts +8 -4
  263. package/src/routes/api/terminals/[id]/resize/+server.ts +8 -4
  264. package/src/routes/api/terminals/[id]/share/+server.ts +98 -0
  265. package/src/routes/api/terminals/[id]/share/auth/+server.ts +81 -0
  266. package/src/routes/api/terminals/[id]/share/status/+server.ts +11 -0
  267. package/src/routes/api/ws-ticket/+server.ts +26 -5
  268. package/src/routes/sos/+page.svelte +195 -0
  269. package/src/routes/sos/[id]/+page.svelte +677 -0
  270. package/src/routes/terminals/[id]/+page.svelte +184 -43
  271. package/build/client/_app/immutable/assets/0.NV8k8wxG.css.br +0 -0
  272. package/build/client/_app/immutable/assets/9.v5KA95xm.css +0 -1
  273. package/build/client/_app/immutable/assets/9.v5KA95xm.css.br +0 -0
  274. package/build/client/_app/immutable/assets/9.v5KA95xm.css.gz +0 -0
  275. package/build/client/_app/immutable/chunks/8lO1IL7u.js.br +0 -0
  276. package/build/client/_app/immutable/chunks/B9WQy_3X.js.br +0 -0
  277. package/build/client/_app/immutable/chunks/B9WQy_3X.js.gz +0 -0
  278. package/build/client/_app/immutable/chunks/BRqaaL5D.js.br +0 -0
  279. package/build/client/_app/immutable/chunks/BRqaaL5D.js.gz +0 -0
  280. package/build/client/_app/immutable/chunks/C5VOyQCG.js.br +0 -0
  281. package/build/client/_app/immutable/chunks/C5VOyQCG.js.gz +0 -0
  282. package/build/client/_app/immutable/chunks/CR6bkGJW.js +0 -6
  283. package/build/client/_app/immutable/chunks/CR6bkGJW.js.br +0 -0
  284. package/build/client/_app/immutable/chunks/CR6bkGJW.js.gz +0 -0
  285. package/build/client/_app/immutable/chunks/C_9BZILB.js.br +0 -0
  286. package/build/client/_app/immutable/chunks/C_9BZILB.js.gz +0 -0
  287. package/build/client/_app/immutable/chunks/Cg3dlX05.js.br +0 -0
  288. package/build/client/_app/immutable/chunks/Cg3dlX05.js.gz +0 -0
  289. package/build/client/_app/immutable/chunks/DJvX78LW.js.br +0 -0
  290. package/build/client/_app/immutable/chunks/DJvX78LW.js.gz +0 -0
  291. package/build/client/_app/immutable/chunks/DYuMZGL5.js.br +0 -0
  292. package/build/client/_app/immutable/chunks/DYuMZGL5.js.gz +0 -0
  293. package/build/client/_app/immutable/chunks/DZQMsHM5.js.br +0 -0
  294. package/build/client/_app/immutable/chunks/DZQMsHM5.js.gz +0 -0
  295. package/build/client/_app/immutable/chunks/DZvnhU_8.js.br +0 -0
  296. package/build/client/_app/immutable/chunks/DZvnhU_8.js.gz +0 -0
  297. package/build/client/_app/immutable/chunks/nWG9RHyB.js +0 -3
  298. package/build/client/_app/immutable/chunks/nWG9RHyB.js.br +0 -0
  299. package/build/client/_app/immutable/chunks/nWG9RHyB.js.gz +0 -0
  300. package/build/client/_app/immutable/entry/app.f46Ko1hu.js +0 -2
  301. package/build/client/_app/immutable/entry/app.f46Ko1hu.js.br +0 -0
  302. package/build/client/_app/immutable/entry/app.f46Ko1hu.js.gz +0 -0
  303. package/build/client/_app/immutable/entry/start.BVDjNnXt.js +0 -1
  304. package/build/client/_app/immutable/entry/start.BVDjNnXt.js.br +0 -2
  305. package/build/client/_app/immutable/entry/start.BVDjNnXt.js.gz +0 -0
  306. package/build/client/_app/immutable/nodes/0.D_9EwVmq.js +0 -7
  307. package/build/client/_app/immutable/nodes/0.D_9EwVmq.js.br +0 -0
  308. package/build/client/_app/immutable/nodes/0.D_9EwVmq.js.gz +0 -0
  309. package/build/client/_app/immutable/nodes/1.C4eFlqSB.js.br +0 -0
  310. package/build/client/_app/immutable/nodes/1.C4eFlqSB.js.gz +0 -0
  311. package/build/client/_app/immutable/nodes/2.CdC092Za.js.br +0 -0
  312. package/build/client/_app/immutable/nodes/2.CdC092Za.js.gz +0 -0
  313. package/build/client/_app/immutable/nodes/3.Dhf4ZWW0.js.br +0 -0
  314. package/build/client/_app/immutable/nodes/3.Dhf4ZWW0.js.gz +0 -0
  315. package/build/client/_app/immutable/nodes/4.BSVqdrrD.js.br +0 -0
  316. package/build/client/_app/immutable/nodes/4.BSVqdrrD.js.gz +0 -0
  317. package/build/client/_app/immutable/nodes/5.Cfj35gpY.js.br +0 -0
  318. package/build/client/_app/immutable/nodes/5.Cfj35gpY.js.gz +0 -0
  319. package/build/client/_app/immutable/nodes/6.B3SEB_li.js.br +0 -0
  320. package/build/client/_app/immutable/nodes/6.B3SEB_li.js.gz +0 -0
  321. package/build/client/_app/immutable/nodes/7.DV8cJ1lX.js.br +0 -0
  322. package/build/client/_app/immutable/nodes/7.DV8cJ1lX.js.gz +0 -0
  323. package/build/client/_app/immutable/nodes/8.Bs362gyb.js.br +0 -0
  324. package/build/client/_app/immutable/nodes/8.Bs362gyb.js.gz +0 -0
  325. package/build/client/_app/immutable/nodes/9.Cf7_3uqT.js +0 -2
  326. package/build/client/_app/immutable/nodes/9.Cf7_3uqT.js.br +0 -0
  327. package/build/client/_app/immutable/nodes/9.Cf7_3uqT.js.gz +0 -0
  328. package/build/server/chunks/1-C4BOGoJY.js +0 -9
  329. package/build/server/chunks/6-D8xbnTSo.js +0 -9
  330. package/build/server/chunks/7-CkVK06S0.js +0 -9
  331. package/build/server/chunks/8-C8qVhrds.js +0 -9
  332. package/build/server/chunks/8-C8qVhrds.js.map +0 -1
  333. package/build/server/chunks/9-fL5zqN0T.js +0 -9
  334. package/build/server/chunks/9-fL5zqN0T.js.map +0 -1
  335. package/build/server/chunks/Button-B5dU-ntz.js.map +0 -1
  336. package/build/server/chunks/Icon-C7Ml3GX6.js.map +0 -1
  337. package/build/server/chunks/Input-CPGO0sbS.js.map +0 -1
  338. package/build/server/chunks/Pill-CcrtCejm.js.map +0 -1
  339. package/build/server/chunks/Shimmer-C5jkvGr1.js.map +0 -1
  340. package/build/server/chunks/_error.svelte-CSIxs-ab.js.map +0 -1
  341. package/build/server/chunks/_layout.svelte-noB4j-v2.js.map +0 -1
  342. package/build/server/chunks/_page.svelte-B6qyh-K-.js.map +0 -1
  343. package/build/server/chunks/_page.svelte-BUkm2304.js.map +0 -1
  344. package/build/server/chunks/_page.svelte-BV0XyYJZ.js.map +0 -1
  345. package/build/server/chunks/_page.svelte-BfB8maoc.js.map +0 -1
  346. package/build/server/chunks/_page.svelte-C60lAagP.js.map +0 -1
  347. package/build/server/chunks/_page.svelte-Dmg-RFCg.js.map +0 -1
  348. package/build/server/chunks/_page.svelte-DnTpPnPR.js.map +0 -1
  349. package/build/server/chunks/_page.svelte-DuzZr5dA.js.map +0 -1
  350. package/build/server/chunks/_server.ts-BA_uWcPw.js.map +0 -1
  351. package/build/server/chunks/_server.ts-Bjbr7glm.js.map +0 -1
  352. package/build/server/chunks/_server.ts-BrqaMMAa.js.map +0 -1
  353. package/build/server/chunks/_server.ts-Bu3s5hfv.js.map +0 -1
  354. package/build/server/chunks/_server.ts-CyjDrcZN.js.map +0 -1
  355. package/build/server/chunks/_server.ts-DZ5naqSL.js.map +0 -1
  356. package/build/server/chunks/_server.ts-DZP2lhaY.js.map +0 -1
  357. package/build/server/chunks/cache-Me3zUAaD.js.map +0 -1
  358. package/build/server/chunks/client2-DDP30_vY.js +0 -7
  359. package/build/server/chunks/events-handler-Dm1mNPQP.js +0 -20
  360. package/build/server/chunks/events-handler-Dm1mNPQP.js.map +0 -1
  361. package/build/server/chunks/pty-manager-DmNSCKAr.js.map +0 -1
  362. package/build/server/chunks/registry-Kcw2UCMv.js.map +0 -1
  363. package/build/server/chunks/root-xvQIR1Bt.js.map +0 -1
  364. package/build/server/chunks/stores-C-LqoonT.js.map +0 -1
  365. /package/build/client/_app/immutable/assets/{8.BhoBXADL.css → 10.BhoBXADL.css} +0 -0
  366. /package/build/client/_app/immutable/assets/{8.BhoBXADL.css.br → 10.BhoBXADL.css.br} +0 -0
  367. /package/build/client/_app/immutable/assets/{8.BhoBXADL.css.gz → 10.BhoBXADL.css.gz} +0 -0
@@ -0,0 +1,98 @@
1
+ // /api/terminals/[id]/share — owner management of a terminal's share.
2
+ // GET: current state. PUT: create/update. DELETE: revoke.
3
+ // All methods require the API key (owners only).
4
+
5
+ import type { ShareConfigRequest, ShareInfoResponse, ShareMode } from '$lib/types';
6
+
7
+ import { validateAuth } from '$lib/modules/server/auth';
8
+ import { ptyManager } from '$lib/modules/server/terminal/pty-manager.js';
9
+ import { hashPassword, shareStore } from '$lib/modules/server/terminal/share-store';
10
+ import { closeGuests } from '$lib/modules/server/ws/guest-registry';
11
+ import { json } from '@sveltejs/kit';
12
+
13
+ import type { RequestHandler } from './$types';
14
+
15
+ const MIN_PASSWORD_LENGTH = 6;
16
+ const MODES: ShareMode[] = ['view', 'control'];
17
+
18
+ function toInfo(terminalId: string): ShareInfoResponse {
19
+ const share = shareStore.getShare(terminalId);
20
+ if (!share) {
21
+ return { active: false, createdAt: null, mode: null, updatedAt: null };
22
+ }
23
+ return { active: true, createdAt: share.createdAt, mode: share.mode, updatedAt: share.updatedAt };
24
+ }
25
+
26
+ export const GET: RequestHandler = ({ params, request }) => {
27
+ const authError = validateAuth(request);
28
+ if (authError) {
29
+ return authError;
30
+ }
31
+ return json(toInfo(params.id));
32
+ };
33
+
34
+ export const PUT: RequestHandler = async ({ params, request }) => {
35
+ const authError = validateAuth(request);
36
+ if (authError) {
37
+ return authError;
38
+ }
39
+ if (!ptyManager.get(params.id)) {
40
+ return json({ error: 'Terminal not found' }, { status: 404 });
41
+ }
42
+
43
+ let body: ShareConfigRequest;
44
+ try {
45
+ body = (await request.json()) as ShareConfigRequest;
46
+ } catch {
47
+ return json({ error: 'Invalid JSON' }, { status: 400 });
48
+ }
49
+ if (!MODES.includes(body.mode)) {
50
+ return json({ error: "mode must be 'view' or 'control'" }, { status: 400 });
51
+ }
52
+
53
+ const existing = shareStore.getShare(params.id);
54
+ const password = typeof body.password === 'string' ? body.password : '';
55
+ if (!existing && password.length < MIN_PASSWORD_LENGTH) {
56
+ return json(
57
+ { error: `password is required (min ${String(MIN_PASSWORD_LENGTH)} chars)` },
58
+ { status: 400 }
59
+ );
60
+ }
61
+ if (password && password.length < MIN_PASSWORD_LENGTH) {
62
+ return json(
63
+ { error: `password must be at least ${String(MIN_PASSWORD_LENGTH)} chars` },
64
+ { status: 400 }
65
+ );
66
+ }
67
+
68
+ const now = Date.now();
69
+ shareStore.setShare({
70
+ createdAt: existing?.createdAt ?? now,
71
+ mode: body.mode,
72
+ // `existing` is guaranteed non-null when password is empty (validated above).
73
+ passwordHash: password ? hashPassword(password) : (existing?.passwordHash ?? ''),
74
+ terminalId: params.id,
75
+ updatedAt: now,
76
+ });
77
+
78
+ // A new password invalidates existing guest sessions; any change to the
79
+ // share forces connected guests to reconnect under the new scope.
80
+ if (password) {
81
+ shareStore.deleteSessions(params.id);
82
+ }
83
+ if (password || existing?.mode !== body.mode) {
84
+ closeGuests(params.id);
85
+ }
86
+
87
+ return json(toInfo(params.id));
88
+ };
89
+
90
+ export const DELETE: RequestHandler = ({ params, request }) => {
91
+ const authError = validateAuth(request);
92
+ if (authError) {
93
+ return authError;
94
+ }
95
+ shareStore.deleteShare(params.id);
96
+ const closed = closeGuests(params.id);
97
+ return json({ closedConnections: closed, success: true });
98
+ };
@@ -0,0 +1,81 @@
1
+ // POST /api/terminals/[id]/share/auth — exchange the share password for a
2
+ // guest session token. Public endpoint; brute-force-limited per IP+terminal.
3
+
4
+ import type { ShareAuthRequest } from '$lib/types';
5
+
6
+ import { shareStore, verifyPassword } from '$lib/modules/server/terminal/share-store';
7
+ import { json } from '@sveltejs/kit';
8
+
9
+ import type { RequestHandler } from './$types';
10
+
11
+ const RATE_LIMIT_WINDOW_MS = 60_000;
12
+ const RATE_LIMIT_MAX = 10;
13
+
14
+ /** Maps "ip:terminalId" -> attempt timestamps (epoch ms). */
15
+ const attempts = new Map<string, number[]>();
16
+
17
+ function checkRateLimit(key: string): boolean {
18
+ const now = Date.now();
19
+ const recent = (attempts.get(key) ?? []).filter((t) => t > now - RATE_LIMIT_WINDOW_MS);
20
+ attempts.set(key, recent);
21
+ if (recent.length >= RATE_LIMIT_MAX) {
22
+ return false;
23
+ }
24
+ recent.push(now);
25
+ return true;
26
+ }
27
+
28
+ // Cleanup stale rate limit entries every 5 minutes
29
+ setInterval(() => {
30
+ const cutoff = Date.now() - RATE_LIMIT_WINDOW_MS;
31
+ for (const [key, timestamps] of attempts) {
32
+ const recent = timestamps.filter((t) => t > cutoff);
33
+ if (recent.length === 0) {
34
+ attempts.delete(key);
35
+ } else {
36
+ attempts.set(key, recent);
37
+ }
38
+ }
39
+ }, 300_000).unref();
40
+
41
+ export const POST: RequestHandler = async (event) => {
42
+ const { params, request } = event;
43
+
44
+ const share = shareStore.getShare(params.id);
45
+ if (!share) {
46
+ return json({ error: 'Not shared' }, { status: 404 });
47
+ }
48
+
49
+ // Behind Cloudflare Tunnel the connecting IP arrives in headers.
50
+ let ip = 'unknown';
51
+ const cfIp = request.headers.get('cf-connecting-ip');
52
+ const fwd = request.headers.get('x-forwarded-for');
53
+ if (cfIp) {
54
+ ip = cfIp;
55
+ } else if (fwd) {
56
+ ip = fwd.split(',')[0].trim();
57
+ } else {
58
+ try {
59
+ ip = event.getClientAddress();
60
+ } catch {
61
+ // Keep 'unknown' — the rate limit still applies per terminal.
62
+ }
63
+ }
64
+
65
+ if (!checkRateLimit(`${ip}:${params.id}`)) {
66
+ return json({ error: 'Too many attempts. Try again in a minute.' }, { status: 429 });
67
+ }
68
+
69
+ let body: ShareAuthRequest;
70
+ try {
71
+ body = (await request.json()) as ShareAuthRequest;
72
+ } catch {
73
+ return json({ error: 'Invalid JSON' }, { status: 400 });
74
+ }
75
+ if (typeof body.password !== 'string' || !verifyPassword(body.password, share.passwordHash)) {
76
+ return json({ error: 'Invalid password' }, { status: 401 });
77
+ }
78
+
79
+ const { expiresAt, token } = shareStore.createSession(params.id);
80
+ return json({ expiresAt, mode: share.mode, token });
81
+ };
@@ -0,0 +1,11 @@
1
+ // GET /api/terminals/[id]/share/status — public probe used by the page to
2
+ // decide whether to show the password gate. Reveals only a boolean.
3
+
4
+ import { shareStore } from '$lib/modules/server/terminal/share-store';
5
+ import { json } from '@sveltejs/kit';
6
+
7
+ import type { RequestHandler } from './$types';
8
+
9
+ export const GET: RequestHandler = ({ params }) => {
10
+ return json({ shared: shareStore.getShare(params.id) !== null });
11
+ };
@@ -8,6 +8,7 @@
8
8
  // Rate limited to 30 requests per minute per API key.
9
9
 
10
10
  import { validateAuth } from '$lib/modules/server/auth';
11
+ import { shareStore } from '$lib/modules/server/terminal/share-store';
11
12
  import { generateTicket } from '$lib/modules/server/ws/ticket-store';
12
13
  import { json } from '@sveltejs/kit';
13
14
 
@@ -63,15 +64,35 @@ setInterval(() => {
63
64
  // ── Endpoint ────────────────────────────────────────────────────────
64
65
 
65
66
  export const POST: RequestHandler = ({ request }) => {
67
+ const bearer = (
68
+ request.headers.get('authorization') ??
69
+ request.headers.get('Authorization') ??
70
+ ''
71
+ )
72
+ .replace(/^Bearer\s+/i, '')
73
+ .trim();
74
+
66
75
  const authError = validateAuth(request);
67
76
  if (authError) {
68
- return authError;
77
+ // Not the API key — maybe a guest share token (issues a scoped ticket).
78
+ const session = bearer ? shareStore.resolveToken(bearer) : null;
79
+ if (!session) {
80
+ return authError;
81
+ }
82
+ if (!checkRateLimit(bearer)) {
83
+ return json(
84
+ { error: 'Rate limit exceeded. Maximum 30 ticket requests per minute.' },
85
+ { status: 429 }
86
+ );
87
+ }
88
+ const ticket = generateTicket({
89
+ readOnly: session.mode === 'view',
90
+ terminalId: session.terminalId,
91
+ });
92
+ return json({ expiresIn: 30, ticket });
69
93
  }
70
94
 
71
- // Extract the API key for rate limiting
72
- const apiKey = (request.headers.get('authorization') ?? '').substring(7).trim();
73
-
74
- if (!checkRateLimit(apiKey)) {
95
+ if (!checkRateLimit(bearer)) {
75
96
  return json(
76
97
  { error: 'Rate limit exceeded. Maximum 30 ticket requests per minute.' },
77
98
  { status: 429 }
@@ -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>