@juspay/shooter 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (327) hide show
  1. package/.claude/hooks/notifier.cjs +1431 -0
  2. package/.claude/settings.json +162 -0
  3. package/README.md +515 -0
  4. package/bin/shooter.cjs +141 -0
  5. package/build/client/_app/immutable/assets/0.CM9Hl6d-.css +1 -0
  6. package/build/client/_app/immutable/assets/0.CM9Hl6d-.css.br +0 -0
  7. package/build/client/_app/immutable/assets/0.CM9Hl6d-.css.gz +0 -0
  8. package/build/client/_app/immutable/assets/2.CAShZ7lQ.css +1 -0
  9. package/build/client/_app/immutable/assets/2.CAShZ7lQ.css.br +1 -0
  10. package/build/client/_app/immutable/assets/2.CAShZ7lQ.css.gz +0 -0
  11. package/build/client/_app/immutable/assets/3.C0uFg0IS.css +1 -0
  12. package/build/client/_app/immutable/assets/3.C0uFg0IS.css.br +0 -0
  13. package/build/client/_app/immutable/assets/3.C0uFg0IS.css.gz +0 -0
  14. package/build/client/_app/immutable/assets/4.cJuCkJKZ.css +1 -0
  15. package/build/client/_app/immutable/assets/4.cJuCkJKZ.css.br +0 -0
  16. package/build/client/_app/immutable/assets/4.cJuCkJKZ.css.gz +0 -0
  17. package/build/client/_app/immutable/assets/5.DRjApZQW.css +1 -0
  18. package/build/client/_app/immutable/assets/5.DRjApZQW.css.br +0 -0
  19. package/build/client/_app/immutable/assets/5.DRjApZQW.css.gz +0 -0
  20. package/build/client/_app/immutable/assets/6.AraZrY8I.css +1 -0
  21. package/build/client/_app/immutable/assets/6.AraZrY8I.css.br +0 -0
  22. package/build/client/_app/immutable/assets/6.AraZrY8I.css.gz +0 -0
  23. package/build/client/_app/immutable/assets/7.BCJ1IuMx.css +1 -0
  24. package/build/client/_app/immutable/assets/7.BCJ1IuMx.css.br +0 -0
  25. package/build/client/_app/immutable/assets/7.BCJ1IuMx.css.gz +0 -0
  26. package/build/client/_app/immutable/assets/ChatView.CsdBAOKx.css +1 -0
  27. package/build/client/_app/immutable/assets/ChatView.CsdBAOKx.css.br +0 -0
  28. package/build/client/_app/immutable/assets/ChatView.CsdBAOKx.css.gz +0 -0
  29. package/build/client/_app/immutable/assets/markdown.B0b5w2tq.css +1 -0
  30. package/build/client/_app/immutable/assets/markdown.B0b5w2tq.css.br +0 -0
  31. package/build/client/_app/immutable/assets/markdown.B0b5w2tq.css.gz +0 -0
  32. package/build/client/_app/immutable/assets/xterm.DFuMZ0ql.css +1 -0
  33. package/build/client/_app/immutable/assets/xterm.DFuMZ0ql.css.br +0 -0
  34. package/build/client/_app/immutable/assets/xterm.DFuMZ0ql.css.gz +0 -0
  35. package/build/client/_app/immutable/chunks/BNJphC1q.js +56 -0
  36. package/build/client/_app/immutable/chunks/BNJphC1q.js.br +0 -0
  37. package/build/client/_app/immutable/chunks/BNJphC1q.js.gz +0 -0
  38. package/build/client/_app/immutable/chunks/BTGVxaYV.js +9 -0
  39. package/build/client/_app/immutable/chunks/BTGVxaYV.js.br +0 -0
  40. package/build/client/_app/immutable/chunks/BTGVxaYV.js.gz +0 -0
  41. package/build/client/_app/immutable/chunks/BlxrFPDK.js +1 -0
  42. package/build/client/_app/immutable/chunks/BlxrFPDK.js.br +0 -0
  43. package/build/client/_app/immutable/chunks/BlxrFPDK.js.gz +0 -0
  44. package/build/client/_app/immutable/chunks/Bvk7mfPM.js +1 -0
  45. package/build/client/_app/immutable/chunks/Bvk7mfPM.js.br +0 -0
  46. package/build/client/_app/immutable/chunks/Bvk7mfPM.js.gz +0 -0
  47. package/build/client/_app/immutable/chunks/CAokzuPQ.js +1 -0
  48. package/build/client/_app/immutable/chunks/CAokzuPQ.js.br +0 -0
  49. package/build/client/_app/immutable/chunks/CAokzuPQ.js.gz +0 -0
  50. package/build/client/_app/immutable/chunks/CGLrx-H5.js +1 -0
  51. package/build/client/_app/immutable/chunks/CGLrx-H5.js.br +0 -0
  52. package/build/client/_app/immutable/chunks/CGLrx-H5.js.gz +0 -0
  53. package/build/client/_app/immutable/chunks/CgCpWzEA.js +1 -0
  54. package/build/client/_app/immutable/chunks/CgCpWzEA.js.br +0 -0
  55. package/build/client/_app/immutable/chunks/CgCpWzEA.js.gz +0 -0
  56. package/build/client/_app/immutable/chunks/Cjwk_cGO.js +6 -0
  57. package/build/client/_app/immutable/chunks/Cjwk_cGO.js.br +0 -0
  58. package/build/client/_app/immutable/chunks/Cjwk_cGO.js.gz +0 -0
  59. package/build/client/_app/immutable/chunks/CtQ8EED1.js +11 -0
  60. package/build/client/_app/immutable/chunks/CtQ8EED1.js.br +0 -0
  61. package/build/client/_app/immutable/chunks/CtQ8EED1.js.gz +0 -0
  62. package/build/client/_app/immutable/chunks/DERQCisl.js +1 -0
  63. package/build/client/_app/immutable/chunks/DERQCisl.js.br +0 -0
  64. package/build/client/_app/immutable/chunks/DERQCisl.js.gz +0 -0
  65. package/build/client/_app/immutable/chunks/DKrg8TQs.js +1 -0
  66. package/build/client/_app/immutable/chunks/DKrg8TQs.js.br +0 -0
  67. package/build/client/_app/immutable/chunks/DKrg8TQs.js.gz +0 -0
  68. package/build/client/_app/immutable/chunks/DLu6yJIZ.js +1 -0
  69. package/build/client/_app/immutable/chunks/DLu6yJIZ.js.br +0 -0
  70. package/build/client/_app/immutable/chunks/DLu6yJIZ.js.gz +0 -0
  71. package/build/client/_app/immutable/chunks/Dkkpz_4D.js +126 -0
  72. package/build/client/_app/immutable/chunks/Dkkpz_4D.js.br +0 -0
  73. package/build/client/_app/immutable/chunks/Dkkpz_4D.js.gz +0 -0
  74. package/build/client/_app/immutable/chunks/DoczjQhA.js +1 -0
  75. package/build/client/_app/immutable/chunks/DoczjQhA.js.br +0 -0
  76. package/build/client/_app/immutable/chunks/DoczjQhA.js.gz +0 -0
  77. package/build/client/_app/immutable/chunks/PPVm8Dsz.js +1 -0
  78. package/build/client/_app/immutable/chunks/PPVm8Dsz.js.br +0 -0
  79. package/build/client/_app/immutable/chunks/PPVm8Dsz.js.gz +0 -0
  80. package/build/client/_app/immutable/chunks/RpcNruLP.js +2 -0
  81. package/build/client/_app/immutable/chunks/RpcNruLP.js.br +0 -0
  82. package/build/client/_app/immutable/chunks/RpcNruLP.js.gz +0 -0
  83. package/build/client/_app/immutable/chunks/a-St0Zwo.js +1 -0
  84. package/build/client/_app/immutable/chunks/a-St0Zwo.js.br +0 -0
  85. package/build/client/_app/immutable/chunks/a-St0Zwo.js.gz +0 -0
  86. package/build/client/_app/immutable/chunks/bo70OQUZ.js +1 -0
  87. package/build/client/_app/immutable/chunks/bo70OQUZ.js.br +0 -0
  88. package/build/client/_app/immutable/chunks/bo70OQUZ.js.gz +0 -0
  89. package/build/client/_app/immutable/entry/app.QvGgdvTI.js +2 -0
  90. package/build/client/_app/immutable/entry/app.QvGgdvTI.js.br +0 -0
  91. package/build/client/_app/immutable/entry/app.QvGgdvTI.js.gz +0 -0
  92. package/build/client/_app/immutable/entry/start.BntDNRMC.js +1 -0
  93. package/build/client/_app/immutable/entry/start.BntDNRMC.js.br +0 -0
  94. package/build/client/_app/immutable/entry/start.BntDNRMC.js.gz +0 -0
  95. package/build/client/_app/immutable/nodes/0.CzkdvJ7j.js +1 -0
  96. package/build/client/_app/immutable/nodes/0.CzkdvJ7j.js.br +0 -0
  97. package/build/client/_app/immutable/nodes/0.CzkdvJ7j.js.gz +0 -0
  98. package/build/client/_app/immutable/nodes/1.MG1QhfrI.js +1 -0
  99. package/build/client/_app/immutable/nodes/1.MG1QhfrI.js.br +0 -0
  100. package/build/client/_app/immutable/nodes/1.MG1QhfrI.js.gz +0 -0
  101. package/build/client/_app/immutable/nodes/2.B4MlOSh6.js +1 -0
  102. package/build/client/_app/immutable/nodes/2.B4MlOSh6.js.br +0 -0
  103. package/build/client/_app/immutable/nodes/2.B4MlOSh6.js.gz +0 -0
  104. package/build/client/_app/immutable/nodes/3.DIwYkjDn.js +3 -0
  105. package/build/client/_app/immutable/nodes/3.DIwYkjDn.js.br +0 -0
  106. package/build/client/_app/immutable/nodes/3.DIwYkjDn.js.gz +0 -0
  107. package/build/client/_app/immutable/nodes/4.D-cIe70D.js +1 -0
  108. package/build/client/_app/immutable/nodes/4.D-cIe70D.js.br +0 -0
  109. package/build/client/_app/immutable/nodes/4.D-cIe70D.js.gz +0 -0
  110. package/build/client/_app/immutable/nodes/5.D7zPRe3L.js +1 -0
  111. package/build/client/_app/immutable/nodes/5.D7zPRe3L.js.br +0 -0
  112. package/build/client/_app/immutable/nodes/5.D7zPRe3L.js.gz +0 -0
  113. package/build/client/_app/immutable/nodes/6.BB7QE48r.js +2 -0
  114. package/build/client/_app/immutable/nodes/6.BB7QE48r.js.br +0 -0
  115. package/build/client/_app/immutable/nodes/6.BB7QE48r.js.gz +0 -0
  116. package/build/client/_app/immutable/nodes/7.D8mqsrZG.js +2 -0
  117. package/build/client/_app/immutable/nodes/7.D8mqsrZG.js.br +0 -0
  118. package/build/client/_app/immutable/nodes/7.D8mqsrZG.js.gz +0 -0
  119. package/build/client/_app/version.json +1 -0
  120. package/build/client/_app/version.json.br +0 -0
  121. package/build/client/_app/version.json.gz +0 -0
  122. package/build/client/app-icon.png +0 -0
  123. package/build/client/apple-touch-icon.png +0 -0
  124. package/build/client/favicon.png +0 -0
  125. package/build/client/favicon.svg +10 -0
  126. package/build/client/favicon.svg.br +0 -0
  127. package/build/client/favicon.svg.gz +0 -0
  128. package/build/client/manifest.webmanifest +1 -0
  129. package/build/client/pwa-192x192.png +0 -0
  130. package/build/client/pwa-512x512.png +0 -0
  131. package/build/client/registerSW.js +1 -0
  132. package/build/client/registerSW.js.br +0 -0
  133. package/build/client/registerSW.js.gz +0 -0
  134. package/build/client/sw.js +222 -0
  135. package/build/client/sw.js.br +0 -0
  136. package/build/client/sw.js.gz +0 -0
  137. package/build/client/workbox-5119daf5.js +3395 -0
  138. package/build/client/workbox-5119daf5.js.br +0 -0
  139. package/build/client/workbox-5119daf5.js.gz +0 -0
  140. package/build/env.js +94 -0
  141. package/build/handler.js +1494 -0
  142. package/build/index.js +345 -0
  143. package/build/pty-holder.cjs +510 -0
  144. package/build/server/chunks/0-q2IUp76Y.js +9 -0
  145. package/build/server/chunks/0-q2IUp76Y.js.map +1 -0
  146. package/build/server/chunks/1-CU50G5wZ.js +9 -0
  147. package/build/server/chunks/1-CU50G5wZ.js.map +1 -0
  148. package/build/server/chunks/2-D01t9s8T.js +9 -0
  149. package/build/server/chunks/2-D01t9s8T.js.map +1 -0
  150. package/build/server/chunks/3-5PUQ04wC.js +9 -0
  151. package/build/server/chunks/3-5PUQ04wC.js.map +1 -0
  152. package/build/server/chunks/4-e7gywnSG.js +9 -0
  153. package/build/server/chunks/4-e7gywnSG.js.map +1 -0
  154. package/build/server/chunks/5-CA1SA6KZ.js +9 -0
  155. package/build/server/chunks/5-CA1SA6KZ.js.map +1 -0
  156. package/build/server/chunks/6-71H221sV.js +9 -0
  157. package/build/server/chunks/6-71H221sV.js.map +1 -0
  158. package/build/server/chunks/7-Bo-vmdyz.js +9 -0
  159. package/build/server/chunks/7-Bo-vmdyz.js.map +1 -0
  160. package/build/server/chunks/_layout.svelte-SFHOxs74.js +132 -0
  161. package/build/server/chunks/_layout.svelte-SFHOxs74.js.map +1 -0
  162. package/build/server/chunks/_page.svelte-B4w-2wD-.js +120 -0
  163. package/build/server/chunks/_page.svelte-B4w-2wD-.js.map +1 -0
  164. package/build/server/chunks/_page.svelte-B_qAXjkh.js +213 -0
  165. package/build/server/chunks/_page.svelte-B_qAXjkh.js.map +1 -0
  166. package/build/server/chunks/_page.svelte-CsF1_TRG.js +50 -0
  167. package/build/server/chunks/_page.svelte-CsF1_TRG.js.map +1 -0
  168. package/build/server/chunks/_page.svelte-DJC6U-P0.js +68 -0
  169. package/build/server/chunks/_page.svelte-DJC6U-P0.js.map +1 -0
  170. package/build/server/chunks/_page.svelte-DQ6HBtsz.js +407 -0
  171. package/build/server/chunks/_page.svelte-DQ6HBtsz.js.map +1 -0
  172. package/build/server/chunks/_page.svelte-LbhhjP21.js +148 -0
  173. package/build/server/chunks/_page.svelte-LbhhjP21.js.map +1 -0
  174. package/build/server/chunks/_server.ts-BL2FGb5Z.js +387 -0
  175. package/build/server/chunks/_server.ts-BL2FGb5Z.js.map +1 -0
  176. package/build/server/chunks/_server.ts-BgdjBZco.js +47 -0
  177. package/build/server/chunks/_server.ts-BgdjBZco.js.map +1 -0
  178. package/build/server/chunks/_server.ts-BihKSdj_.js +59 -0
  179. package/build/server/chunks/_server.ts-BihKSdj_.js.map +1 -0
  180. package/build/server/chunks/_server.ts-BjOJsoy4.js +63 -0
  181. package/build/server/chunks/_server.ts-BjOJsoy4.js.map +1 -0
  182. package/build/server/chunks/_server.ts-C29xzfaw.js +77 -0
  183. package/build/server/chunks/_server.ts-C29xzfaw.js.map +1 -0
  184. package/build/server/chunks/_server.ts-CPa6DgIt.js +71 -0
  185. package/build/server/chunks/_server.ts-CPa6DgIt.js.map +1 -0
  186. package/build/server/chunks/_server.ts-CbDRDIoP.js +36 -0
  187. package/build/server/chunks/_server.ts-CbDRDIoP.js.map +1 -0
  188. package/build/server/chunks/_server.ts-Cl1OEWL4.js +54 -0
  189. package/build/server/chunks/_server.ts-Cl1OEWL4.js.map +1 -0
  190. package/build/server/chunks/_server.ts-ColfDHW8.js +60 -0
  191. package/build/server/chunks/_server.ts-ColfDHW8.js.map +1 -0
  192. package/build/server/chunks/_server.ts-Cv_OrRuL.js +494 -0
  193. package/build/server/chunks/_server.ts-Cv_OrRuL.js.map +1 -0
  194. package/build/server/chunks/_server.ts-D4MNi4cD.js +25 -0
  195. package/build/server/chunks/_server.ts-D4MNi4cD.js.map +1 -0
  196. package/build/server/chunks/_server.ts-DRVbgm6k.js +125 -0
  197. package/build/server/chunks/_server.ts-DRVbgm6k.js.map +1 -0
  198. package/build/server/chunks/_server.ts-DfajWaqh.js +39 -0
  199. package/build/server/chunks/_server.ts-DfajWaqh.js.map +1 -0
  200. package/build/server/chunks/_server.ts-y9-WYDMa.js +35 -0
  201. package/build/server/chunks/_server.ts-y9-WYDMa.js.map +1 -0
  202. package/build/server/chunks/auth-CEgFis71.js +32 -0
  203. package/build/server/chunks/auth-CEgFis71.js.map +1 -0
  204. package/build/server/chunks/client-CxCatAKr.js +255 -0
  205. package/build/server/chunks/client-CxCatAKr.js.map +1 -0
  206. package/build/server/chunks/error.svelte-BqdwMWdK.js +26 -0
  207. package/build/server/chunks/error.svelte-BqdwMWdK.js.map +1 -0
  208. package/build/server/chunks/exports-CJ0Q5XmL.js +4081 -0
  209. package/build/server/chunks/exports-CJ0Q5XmL.js.map +1 -0
  210. package/build/server/chunks/index2-DAxIoAO-.js +36 -0
  211. package/build/server/chunks/index2-DAxIoAO-.js.map +1 -0
  212. package/build/server/chunks/jsonl-parser-dmZU_Hyu.js +137 -0
  213. package/build/server/chunks/jsonl-parser-dmZU_Hyu.js.map +1 -0
  214. package/build/server/chunks/library-apns-BHxLmuIx.js +104 -0
  215. package/build/server/chunks/library-apns-BHxLmuIx.js.map +1 -0
  216. package/build/server/chunks/markdown-Bxrl3cCF.js +1241 -0
  217. package/build/server/chunks/markdown-Bxrl3cCF.js.map +1 -0
  218. package/build/server/chunks/pending-requests-D8UiTw7L.js +44 -0
  219. package/build/server/chunks/pending-requests-D8UiTw7L.js.map +1 -0
  220. package/build/server/chunks/pty-manager-C0FhBiVq.js +1697 -0
  221. package/build/server/chunks/pty-manager-C0FhBiVq.js.map +1 -0
  222. package/build/server/chunks/shared-server-BDY8jh20.js +200 -0
  223. package/build/server/chunks/shared-server-BDY8jh20.js.map +1 -0
  224. package/build/server/chunks/stores-D0HorpgL.js +36 -0
  225. package/build/server/chunks/stores-D0HorpgL.js.map +1 -0
  226. package/build/server/index.js +6466 -0
  227. package/build/server/index.js.map +1 -0
  228. package/build/server/manifest.js +184 -0
  229. package/build/server/manifest.js.map +1 -0
  230. package/build/shims.js +32 -0
  231. package/package.json +94 -0
  232. package/scripts/clipboard-shims/wl-paste +48 -0
  233. package/scripts/clipboard-shims/xclip +31 -0
  234. package/scripts/install.sh +477 -0
  235. package/scripts/setup-node-pty.sh +63 -0
  236. package/scripts/setup.cjs +571 -0
  237. package/scripts/test-runner.ts +243 -0
  238. package/scripts/vercel-env-commands.sh +60 -0
  239. package/server.ts +139 -0
  240. package/src/app.css +1835 -0
  241. package/src/app.d.ts +31 -0
  242. package/src/app.html +24 -0
  243. package/src/generated/types/APN.ts +305 -0
  244. package/src/generated/types/CLI.ts +52 -0
  245. package/src/generated/types/JWT.ts +92 -0
  246. package/src/generated/types/Terminal.ts +2736 -0
  247. package/src/generated/types/index.ts +6 -0
  248. package/src/lib/assets/icons/alert-triangle.svg +5 -0
  249. package/src/lib/assets/icons/bell.svg +4 -0
  250. package/src/lib/assets/icons/check-circle.svg +4 -0
  251. package/src/lib/assets/icons/file.svg +4 -0
  252. package/src/lib/assets/icons/folder.svg +3 -0
  253. package/src/lib/assets/icons/play.svg +3 -0
  254. package/src/lib/assets/icons/refresh.svg +4 -0
  255. package/src/lib/assets/icons/settings.svg +4 -0
  256. package/src/lib/assets/icons/terminal.svg +1 -0
  257. package/src/lib/assets/icons/tool.svg +3 -0
  258. package/src/lib/assets/icons/x-circle.svg +5 -0
  259. package/src/lib/modules/client/common/Card.svelte +26 -0
  260. package/src/lib/modules/client/common/EmptyState.svelte +36 -0
  261. package/src/lib/modules/client/common/Icon.svelte +61 -0
  262. package/src/lib/modules/client/common/StatusBadge.svelte +38 -0
  263. package/src/lib/modules/client/common/cache.ts +31 -0
  264. package/src/lib/modules/client/common/config-guard.ts +18 -0
  265. package/src/lib/modules/client/common/index.ts +12 -0
  266. package/src/lib/modules/client/common/markdown.ts +23 -0
  267. package/src/lib/modules/client/common/native-bridge.ts +50 -0
  268. package/src/lib/modules/client/common/time.ts +22 -0
  269. package/src/lib/modules/client/common/tool-title.ts +28 -0
  270. package/src/lib/modules/client/terminal/ChatView.svelte +400 -0
  271. package/src/lib/modules/client/terminal/CommandPalette.svelte +60 -0
  272. package/src/lib/modules/client/terminal/ConnectionStatus.svelte +99 -0
  273. package/src/lib/modules/client/terminal/LaunchSheet.svelte +294 -0
  274. package/src/lib/modules/client/terminal/QuickKeys.svelte +71 -0
  275. package/src/lib/modules/client/terminal/ShortcutsHelp.svelte +79 -0
  276. package/src/lib/modules/client/terminal/keyboard-shortcuts.ts +70 -0
  277. package/src/lib/modules/client/terminal/xterm-wrapper.ts +243 -0
  278. package/src/lib/modules/server/apn/library-apns.ts +137 -0
  279. package/src/lib/modules/server/apn/notification-history.ts +35 -0
  280. package/src/lib/modules/server/apn/notification-sessions.ts +117 -0
  281. package/src/lib/modules/server/apn/pending-requests.ts +65 -0
  282. package/src/lib/modules/server/apn/types.ts +51 -0
  283. package/src/lib/modules/server/auth.ts +34 -0
  284. package/src/lib/modules/server/cli/index.ts +79 -0
  285. package/src/lib/modules/server/cli/runner.ts +162 -0
  286. package/src/lib/modules/server/fcm/fcm-service.ts +72 -0
  287. package/src/lib/modules/server/sessions/jsonl-parser.ts +197 -0
  288. package/src/lib/modules/server/sessions/jsonl-reader.ts +301 -0
  289. package/src/lib/modules/server/sessions/opencode-reader.ts +264 -0
  290. package/src/lib/modules/server/sessions/types.ts +53 -0
  291. package/src/lib/modules/server/terminal/holder-client.ts +273 -0
  292. package/src/lib/modules/server/terminal/opencode-watcher.ts +661 -0
  293. package/src/lib/modules/server/terminal/pty-holder.cjs +510 -0
  294. package/src/lib/modules/server/terminal/pty-manager.ts +1012 -0
  295. package/src/lib/modules/server/terminal/session-watcher.ts +320 -0
  296. package/src/lib/modules/server/terminal/terminal-store.ts +198 -0
  297. package/src/lib/modules/server/ws/events-handler.ts +73 -0
  298. package/src/lib/modules/server/ws/keepalive.ts +108 -0
  299. package/src/lib/modules/server/ws/server.ts +93 -0
  300. package/src/lib/modules/server/ws/session-handler.ts +462 -0
  301. package/src/lib/modules/server/ws/terminal-handler.ts +197 -0
  302. package/src/lib/modules/server/ws/ticket-store.ts +58 -0
  303. package/src/lib/theme.css +529 -0
  304. package/src/lib/types/config.ts +6 -0
  305. package/src/routes/+layout.svelte +218 -0
  306. package/src/routes/+page.svelte +261 -0
  307. package/src/routes/api/debug/+server.ts +33 -0
  308. package/src/routes/api/device-token/+server.ts +85 -0
  309. package/src/routes/api/health/+server.ts +100 -0
  310. package/src/routes/api/notify/+server.ts +418 -0
  311. package/src/routes/api/qr-config/+server.ts +45 -0
  312. package/src/routes/api/response/+server.ts +73 -0
  313. package/src/routes/api/sessions/+server.ts +120 -0
  314. package/src/routes/api/terminals/+server.ts +141 -0
  315. package/src/routes/api/terminals/[id]/+server.ts +75 -0
  316. package/src/routes/api/terminals/[id]/paste-image/+server.ts +61 -0
  317. package/src/routes/api/terminals/[id]/resize/+server.ts +60 -0
  318. package/src/routes/api/webhook/+server.ts +42 -0
  319. package/src/routes/api/ws-status/+server.ts +23 -0
  320. package/src/routes/api/ws-ticket/+server.ts +86 -0
  321. package/src/routes/config/+page.svelte +600 -0
  322. package/src/routes/project/+page.svelte +274 -0
  323. package/src/routes/session/[id]/+page.svelte +434 -0
  324. package/src/routes/terminals/+page.svelte +618 -0
  325. package/src/routes/terminals/[id]/+page.svelte +968 -0
  326. package/svelte.config.js +18 -0
  327. package/tsconfig.json +14 -0
@@ -0,0 +1,600 @@
1
+ <script lang="ts">
2
+ import type { ShooterConfig } from '$lib/types/config';
3
+
4
+ import { browser } from '$app/environment';
5
+ import { goto } from '$app/navigation';
6
+ import { Banner, Button, Input, Stepper } from '@juspay/svelte-ui-components';
7
+ import { Card, hasScanner, Icon, isShooterConfig, scanQR } from '$lib/modules/client/common';
8
+ import { onMount } from 'svelte';
9
+
10
+ let serverUrl = $state('');
11
+ let apiKey = $state('');
12
+ let deviceToken = $state('');
13
+ let result = $state('');
14
+ let loading = $state(false);
15
+ let statusType = $state<'' | 'error' | 'success' | 'warning'>('');
16
+ let isNativeApp = $state(false);
17
+ let qrDataUrl = $state('');
18
+ let qrServerUrl = $state('');
19
+ let qrLoading = $state(false);
20
+ let qrError = $state('');
21
+ let canScan = $state(false);
22
+ let bridgeCheckDone = $state(false);
23
+ let scanLoading = $state(false);
24
+
25
+ async function fetchQrCode(): Promise<void> {
26
+ if (!apiKey.trim()) {
27
+ qrError = 'Save an API key first to generate a QR code';
28
+ return;
29
+ }
30
+
31
+ qrLoading = true;
32
+ qrError = '';
33
+
34
+ try {
35
+ const response = await fetch('/api/qr-config', {
36
+ headers: { Authorization: `Bearer ${apiKey.trim()}` },
37
+ });
38
+
39
+ if (!response.ok) {
40
+ const data = (await response.json()) as { error?: string };
41
+ qrError = data.error || 'Failed to generate QR code';
42
+ return;
43
+ }
44
+
45
+ const data = (await response.json()) as { dataUrl: string; serverUrl: string };
46
+ qrDataUrl = data.dataUrl;
47
+ qrServerUrl = data.serverUrl;
48
+ } catch (error) {
49
+ const err = error as Error;
50
+ qrError = `Network error: ${err.message}`;
51
+ } finally {
52
+ qrLoading = false;
53
+ }
54
+ }
55
+
56
+ async function handleScanQR(): Promise<void> {
57
+ scanLoading = true;
58
+ result = '';
59
+ statusType = '';
60
+
61
+ try {
62
+ // scanQR() returns the raw scanned string (JSON with serverUrl + apiKey)
63
+ // Throws on cancel or error
64
+ const scannedData = await scanQR();
65
+ try {
66
+ const config = JSON.parse(scannedData) as Record<string, unknown>;
67
+ if (typeof config.serverUrl === 'string' && config.serverUrl) {
68
+ serverUrl = config.serverUrl;
69
+ }
70
+ if (typeof config.apiKey === 'string' && config.apiKey) {
71
+ apiKey = config.apiKey;
72
+ }
73
+ await saveConfiguration();
74
+ result = 'Configuration updated from QR code';
75
+ statusType = 'success';
76
+ } catch {
77
+ result = 'Invalid QR code data';
78
+ statusType = 'error';
79
+ }
80
+ } catch (error) {
81
+ const err = error as Error;
82
+ if (err.message !== 'cancelled') {
83
+ result = `Scanner error: ${err.message}`;
84
+ statusType = 'error';
85
+ }
86
+ } finally {
87
+ scanLoading = false;
88
+ }
89
+ }
90
+
91
+ function getNativeBridge(): {
92
+ getConfig(): string;
93
+ saveConfig(json: string): void;
94
+ getFcmToken(): string;
95
+ getPlatform(): string;
96
+ } | null {
97
+ if (typeof window !== 'undefined' && 'ShooterBridge' in window) {
98
+ return (window as Record<string, unknown>).ShooterBridge as {
99
+ getConfig(): string;
100
+ saveConfig(json: string): void;
101
+ getFcmToken(): string;
102
+ getPlatform(): string;
103
+ };
104
+ }
105
+ return null;
106
+ }
107
+
108
+ function getDefaultServerUrl(): string {
109
+ return window.location.origin;
110
+ }
111
+
112
+ onMount(() => {
113
+ if (browser) {
114
+ try {
115
+ serverUrl = getDefaultServerUrl();
116
+ const saved = localStorage.getItem('shooter_config');
117
+ if (saved) {
118
+ const parsed: unknown = JSON.parse(saved);
119
+ if (isShooterConfig(parsed)) {
120
+ if (parsed.apiKey) {
121
+ apiKey = parsed.apiKey;
122
+ }
123
+ if (parsed.deviceToken) {
124
+ deviceToken = parsed.deviceToken;
125
+ }
126
+ if (parsed.serverUrl) {
127
+ serverUrl = parsed.serverUrl;
128
+ }
129
+ } else {
130
+ localStorage.removeItem('shooter_config');
131
+ }
132
+ }
133
+ } catch {
134
+ // No saved configuration — expected on first visit
135
+ }
136
+
137
+ const bridge = getNativeBridge();
138
+ if (bridge) {
139
+ isNativeApp = true;
140
+ try {
141
+ const nativeConfig = JSON.parse(bridge.getConfig());
142
+ // If native has config, use it (native is source of truth for credentials)
143
+ if (nativeConfig.serverUrl && !serverUrl) {
144
+ serverUrl = nativeConfig.serverUrl;
145
+ }
146
+ if (nativeConfig.apiKey) {
147
+ apiKey = nativeConfig.apiKey;
148
+ }
149
+ // Auto-populate device token with FCM token
150
+ const fcmToken = bridge.getFcmToken();
151
+ if (fcmToken) {
152
+ deviceToken = fcmToken;
153
+ }
154
+ } catch {
155
+ // Bridge communication failed
156
+ }
157
+ }
158
+
159
+ canScan = hasScanner();
160
+ bridgeCheckDone = true;
161
+
162
+ // The native bridge may be injected after SvelteKit hydration.
163
+ // Re-check periodically for a short window to catch late injection.
164
+ if (!canScan) {
165
+ const recheckInterval = setInterval(() => {
166
+ canScan = hasScanner();
167
+ if (canScan) {
168
+ clearInterval(recheckInterval);
169
+ }
170
+ }, 200);
171
+ // Stop checking after 3 seconds
172
+ setTimeout(() => clearInterval(recheckInterval), 3000);
173
+ }
174
+ }
175
+ });
176
+
177
+ async function saveConfiguration(): Promise<void> {
178
+ loading = true;
179
+ result = '';
180
+ statusType = '';
181
+
182
+ try {
183
+ const trimmedUrl = serverUrl.trim().replace(/\/+$/, '');
184
+ localStorage.setItem(
185
+ 'shooter_config',
186
+ JSON.stringify({
187
+ apiKey: apiKey.trim(),
188
+ deviceToken: deviceToken.trim(),
189
+ lastUpdated: Date.now(),
190
+ serverUrl: trimmedUrl || getDefaultServerUrl(),
191
+ } satisfies ShooterConfig)
192
+ );
193
+
194
+ const bridge = getNativeBridge();
195
+ if (bridge) {
196
+ bridge.saveConfig(
197
+ JSON.stringify({
198
+ serverUrl: trimmedUrl || getDefaultServerUrl(),
199
+ apiKey: apiKey.trim(),
200
+ })
201
+ );
202
+ }
203
+
204
+ const response = await fetch('/api/health');
205
+
206
+ if (response.ok) {
207
+ result = 'Configuration saved successfully';
208
+ statusType = 'success';
209
+ } else {
210
+ result = 'Configuration saved but system health check failed';
211
+ statusType = 'warning';
212
+ }
213
+ } catch (error) {
214
+ const err = error as Error;
215
+ result = `Configuration failed: ${err.message}`;
216
+ statusType = 'error';
217
+ }
218
+
219
+ loading = false;
220
+ }
221
+
222
+ async function testConfiguration(): Promise<void> {
223
+ if (!apiKey.trim()) {
224
+ result = 'API key is required for testing';
225
+ statusType = 'error';
226
+ return;
227
+ }
228
+
229
+ loading = true;
230
+ result = '';
231
+ statusType = '';
232
+
233
+ try {
234
+ const testPayload: {
235
+ data: Record<string, unknown>;
236
+ deviceToken?: string;
237
+ message: string;
238
+ title: string;
239
+ } = {
240
+ data: { source: 'config-test', timestamp: Date.now() },
241
+ message: `Configuration test at ${new Date().toLocaleTimeString()}`,
242
+ title: 'Configuration Test',
243
+ };
244
+
245
+ if (deviceToken.trim()) {
246
+ testPayload.deviceToken = deviceToken.trim();
247
+ }
248
+
249
+ const response = await fetch('/api/notify', {
250
+ body: JSON.stringify(testPayload),
251
+ headers: {
252
+ Authorization: `Bearer ${apiKey}`,
253
+ 'Content-Type': 'application/json',
254
+ },
255
+ method: 'POST',
256
+ });
257
+
258
+ const data = await response.json();
259
+
260
+ if (response.ok) {
261
+ result = 'Test notification sent successfully';
262
+ statusType = 'success';
263
+ } else {
264
+ result = `Test failed: ${data.error || 'Unknown error'}`;
265
+ statusType = 'error';
266
+ }
267
+ } catch (error) {
268
+ const err = error as Error;
269
+ result = `Network error: ${err.message}`;
270
+ statusType = 'error';
271
+ }
272
+
273
+ loading = false;
274
+ }
275
+
276
+ function clearConfiguration(): void {
277
+ if (browser) {
278
+ localStorage.removeItem('shooter_config');
279
+ serverUrl = typeof window !== 'undefined' ? window.location.origin : '';
280
+ apiKey = '';
281
+ deviceToken = '';
282
+ result = 'Configuration cleared';
283
+ statusType = 'success';
284
+
285
+ const bridge = getNativeBridge();
286
+ if (bridge) {
287
+ bridge.saveConfig(JSON.stringify({ serverUrl: '', apiKey: '' }));
288
+ }
289
+ }
290
+ }
291
+ </script>
292
+
293
+ <svelte:head>
294
+ <title>Settings - Shooter</title>
295
+ <meta name="description" content="Configure notification system settings" />
296
+ </svelte:head>
297
+
298
+ <main class="main">
299
+ <div class="settings-container">
300
+ <div class="page-header">
301
+ <h1 class="page-title">Settings</h1>
302
+ <p class="page-description">Configure your API credentials and notification preferences</p>
303
+ </div>
304
+
305
+ <div class="settings-grid">
306
+ <section class="settings-section">
307
+ <Card title="Server Configuration" description="Configure server connection and credentials">
308
+ <Input
309
+ name="serverUrl"
310
+ label="Server URL"
311
+ bind:value={serverUrl}
312
+ dataType="text"
313
+ placeholder="https://shooter.breezehq.dev"
314
+ infoMessage="Base URL of your Shooter server. Apps will reload with this URL on next launch."
315
+ />
316
+
317
+ <Input
318
+ name="apiKey"
319
+ label="API Key"
320
+ bind:value={apiKey}
321
+ dataType="password"
322
+ placeholder="Enter your API key"
323
+ infoMessage="Required for sending notifications"
324
+ />
325
+
326
+ <Input
327
+ name="deviceToken"
328
+ label="Device Token"
329
+ bind:value={deviceToken}
330
+ dataType="text"
331
+ placeholder={isNativeApp ? 'Waiting for token...' : '64-character hex string'}
332
+ infoMessage={isNativeApp && deviceToken ? 'Auto-detected from app' : 'Device token from app registration'}
333
+ classes="input-mono"
334
+ />
335
+ </Card>
336
+
337
+ {#if result}
338
+ {@const bannerIcon = statusType === 'success' ? 'check-circle' : statusType === 'error' ? 'x-circle' : 'alert-triangle'}
339
+ <Banner text={result} classes="banner-{statusType || 'info'}">
340
+ {#snippet icon()}
341
+ <Icon name={bannerIcon} size={16} />
342
+ {/snippet}
343
+ </Banner>
344
+ {/if}
345
+
346
+ <div class="button-group">
347
+ <Button
348
+ classes="btn-secondary"
349
+ onclick={testConfiguration}
350
+ disabled={loading || !apiKey.trim()}
351
+ >
352
+ {#snippet children()}
353
+ {#if !loading}
354
+ <Icon name="play" size={14} />
355
+ {/if}
356
+ Test Connection
357
+ {/snippet}
358
+ </Button>
359
+ <Button
360
+ classes="btn-primary"
361
+ onclick={saveConfiguration}
362
+ disabled={loading || !apiKey.trim()}
363
+ showLoader={loading}
364
+ text="Save Changes"
365
+ />
366
+ </div>
367
+ </section>
368
+
369
+ <aside class="settings-sidebar">
370
+ <Card title="Setup Guide">
371
+ <Stepper
372
+ steps={[
373
+ { label: 'Get API Key' },
374
+ { label: 'Find Device Token' },
375
+ { label: 'Test Connection' },
376
+ ]}
377
+ currentStepIndex={apiKey.trim() && deviceToken.trim() ? 2 : apiKey.trim() ? 1 : 0}
378
+ classes="setup-stepper"
379
+ />
380
+ </Card>
381
+
382
+ <Card title="Mobile App Setup" description={canScan ? 'Scan a QR code to connect' : 'Scan to connect your mobile app'}>
383
+ <div class="qr-section">
384
+ {#if canScan}
385
+ <p class="qr-description">
386
+ Scan the QR code shown on your server's settings page to auto-configure
387
+ the connection.
388
+ </p>
389
+ <Button classes="btn-secondary btn-sm" onclick={handleScanQR} disabled={scanLoading}
390
+ text={scanLoading ? 'Scanning...' : 'Scan QR Code'}
391
+ />
392
+ {:else}
393
+ {#if qrDataUrl}
394
+ <div class="qr-container">
395
+ <img src={qrDataUrl} alt="QR code for mobile app pairing" class="qr-image" />
396
+ </div>
397
+ <p class="qr-hint">
398
+ Scan this QR code with the Shooter iOS or Android app to connect.
399
+ </p>
400
+ {#if qrServerUrl}
401
+ <p class="qr-server-url">
402
+ Server: <code>{qrServerUrl}</code>
403
+ </p>
404
+ {/if}
405
+ {:else if qrLoading}
406
+ <div class="qr-placeholder">
407
+ <p class="qr-loading-text">Generating QR code...</p>
408
+ </div>
409
+ {:else if qrError}
410
+ <p class="qr-error">{qrError}</p>
411
+ {:else}
412
+ <p class="qr-description">
413
+ Generate a QR code containing your server URL and API key. Your mobile app can scan
414
+ it to auto-configure the connection.
415
+ </p>
416
+ {/if}
417
+ <Button classes="btn-secondary btn-sm" onclick={fetchQrCode} disabled={qrLoading || !apiKey.trim()}
418
+ text={qrDataUrl ? 'Regenerate QR Code' : 'Generate QR Code'}
419
+ />
420
+ {/if}
421
+ </div>
422
+ </Card>
423
+
424
+ <Card title="Danger Zone">
425
+ <p class="danger-description">Clear all saved configuration data from this device.</p>
426
+ <Button classes="btn-danger btn-sm" onclick={clearConfiguration}
427
+ text="Clear Configuration"
428
+ />
429
+ </Card>
430
+ </aside>
431
+ </div>
432
+ </div>
433
+ </main>
434
+
435
+ <style>
436
+ .settings-container {
437
+ max-width: 900px;
438
+ margin: 0 auto;
439
+ padding-bottom: 80px;
440
+ }
441
+
442
+ .page-header {
443
+ margin-bottom: var(--space-8);
444
+ }
445
+
446
+ .page-title {
447
+ font-size: var(--text-2xl);
448
+ font-weight: 600;
449
+ letter-spacing: -0.03em;
450
+ color: var(--text-primary);
451
+ margin-bottom: var(--space-1);
452
+ }
453
+
454
+ .page-description {
455
+ font-size: var(--text-sm);
456
+ color: var(--text-secondary);
457
+ }
458
+
459
+ .settings-grid {
460
+ display: grid;
461
+ grid-template-columns: 1fr 320px;
462
+ gap: var(--space-6);
463
+ align-items: start;
464
+ }
465
+
466
+ .settings-section {
467
+ display: flex;
468
+ flex-direction: column;
469
+ gap: var(--space-4);
470
+ }
471
+
472
+ .settings-sidebar {
473
+ display: flex;
474
+ flex-direction: column;
475
+ gap: var(--space-4);
476
+ }
477
+
478
+ .button-group {
479
+ display: flex;
480
+ gap: var(--space-3);
481
+ justify-content: flex-end;
482
+ }
483
+
484
+ :global(.setup-stepper) {
485
+ --container-flex-direction: column;
486
+ --step-flex-direction: row;
487
+ --step-index-container-height: 24px;
488
+ --step-index-container-width: 24px;
489
+ --step-index-font-size: var(--text-xs);
490
+ --step-text-font-size: var(--text-sm);
491
+ --step-text-margin: 0 0 0 var(--space-3);
492
+ --separator-display: none;
493
+ --step-text-active-color: var(--text-primary);
494
+ --step-text-completed-color: var(--ds-green-700);
495
+ --step-index-container-active-background-color: var(--component-bg-active);
496
+ --step-index-container-completed-background-color: var(--ds-green-700);
497
+ --step-index-container-background-color: var(--component-bg-active);
498
+ --step-index-color: var(--text-secondary);
499
+ gap: var(--space-3);
500
+ }
501
+
502
+ .qr-section {
503
+ display: flex;
504
+ flex-direction: column;
505
+ gap: var(--space-3);
506
+ }
507
+
508
+ .qr-container {
509
+ display: flex;
510
+ justify-content: center;
511
+ padding: var(--space-3);
512
+ background: var(--ds-background-200);
513
+ border: 1px solid var(--border);
514
+ border-radius: var(--radius-lg);
515
+ }
516
+
517
+ .qr-image {
518
+ width: 200px;
519
+ height: 200px;
520
+ border-radius: var(--radius-sm);
521
+ image-rendering: pixelated;
522
+ }
523
+
524
+ .qr-placeholder {
525
+ display: flex;
526
+ align-items: center;
527
+ justify-content: center;
528
+ height: 200px;
529
+ background: var(--ds-background-200);
530
+ border: 1px dashed var(--border);
531
+ border-radius: var(--radius-lg);
532
+ }
533
+
534
+ .qr-loading-text {
535
+ font-size: var(--text-sm);
536
+ color: var(--text-tertiary);
537
+ }
538
+
539
+ .qr-hint {
540
+ font-size: var(--text-xs);
541
+ color: var(--text-secondary);
542
+ line-height: var(--leading-relaxed);
543
+ text-align: center;
544
+ }
545
+
546
+ .qr-server-url {
547
+ font-size: var(--text-xs);
548
+ color: var(--text-tertiary);
549
+ text-align: center;
550
+ }
551
+
552
+ .qr-server-url code {
553
+ font-family: var(--font-mono);
554
+ font-size: var(--text-xs);
555
+ color: var(--text-secondary);
556
+ background: var(--component-bg);
557
+ padding: 1px 4px;
558
+ border-radius: var(--radius-sm);
559
+ }
560
+
561
+ .qr-description {
562
+ font-size: var(--text-sm);
563
+ color: var(--text-secondary);
564
+ line-height: var(--leading-relaxed);
565
+ }
566
+
567
+ .qr-error {
568
+ font-size: var(--text-sm);
569
+ color: var(--ds-red-900);
570
+ line-height: var(--leading-relaxed);
571
+ }
572
+
573
+ .settings-sidebar :global(.card):last-child {
574
+ border-color: rgba(217, 48, 54, 0.3);
575
+ }
576
+
577
+ .danger-description {
578
+ font-size: var(--text-sm);
579
+ color: var(--text-secondary);
580
+ margin-bottom: var(--space-4);
581
+ line-height: var(--leading-relaxed);
582
+ padding-left: var(--space-3);
583
+ border-left: 3px solid rgba(217, 48, 54, 0.5);
584
+ }
585
+
586
+ @media (max-width: 768px) {
587
+ .settings-grid {
588
+ grid-template-columns: 1fr;
589
+ }
590
+
591
+ .button-group {
592
+ flex-direction: column;
593
+ }
594
+
595
+ .button-group :global(button),
596
+ .button-group :global(.button-container) {
597
+ width: 100%;
598
+ }
599
+ }
600
+ </style>