@juspay/shooter 1.19.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 (206) hide show
  1. package/build/client/_app/immutable/assets/11.F10lvwyh.css +1 -0
  2. package/build/client/_app/immutable/assets/11.F10lvwyh.css.br +0 -0
  3. package/build/client/_app/immutable/assets/11.F10lvwyh.css.gz +0 -0
  4. package/build/client/_app/immutable/chunks/C_YNQL8b.js +3 -0
  5. package/build/client/_app/immutable/chunks/C_YNQL8b.js.br +0 -0
  6. package/build/client/_app/immutable/chunks/C_YNQL8b.js.gz +0 -0
  7. package/build/client/_app/immutable/chunks/{DA4Zt9Me.js → DIZ3Qst5.js} +1 -1
  8. package/build/client/_app/immutable/chunks/DIZ3Qst5.js.br +0 -0
  9. package/build/client/_app/immutable/chunks/{DA4Zt9Me.js.gz → DIZ3Qst5.js.gz} +0 -0
  10. package/build/client/_app/immutable/chunks/{DCDL_9ys.js → DT4H19pV.js} +1 -1
  11. package/build/client/_app/immutable/chunks/DT4H19pV.js.br +0 -0
  12. package/build/client/_app/immutable/chunks/DT4H19pV.js.gz +0 -0
  13. package/build/client/_app/immutable/chunks/J5-Cr5oR.js +6 -0
  14. package/build/client/_app/immutable/chunks/J5-Cr5oR.js.br +0 -0
  15. package/build/client/_app/immutable/chunks/J5-Cr5oR.js.gz +0 -0
  16. package/build/client/_app/immutable/entry/{app.D4TXlu7A.js → app.Bd-DfeJi.js} +2 -2
  17. package/build/client/_app/immutable/entry/app.Bd-DfeJi.js.br +0 -0
  18. package/build/client/_app/immutable/entry/app.Bd-DfeJi.js.gz +0 -0
  19. package/build/client/_app/immutable/entry/start.evvp4tX7.js +1 -0
  20. package/build/client/_app/immutable/entry/start.evvp4tX7.js.br +2 -0
  21. package/build/client/_app/immutable/entry/start.evvp4tX7.js.gz +0 -0
  22. package/build/client/_app/immutable/nodes/{0.1zylwAPT.js → 0.Bl-1LQWM.js} +1 -1
  23. package/build/client/_app/immutable/nodes/0.Bl-1LQWM.js.br +0 -0
  24. package/build/client/_app/immutable/nodes/0.Bl-1LQWM.js.gz +0 -0
  25. package/build/client/_app/immutable/nodes/{1.BVnLUSs-.js → 1.DT4dq6Ay.js} +1 -1
  26. package/build/client/_app/immutable/nodes/1.DT4dq6Ay.js.br +0 -0
  27. package/build/client/_app/immutable/nodes/1.DT4dq6Ay.js.gz +0 -0
  28. package/build/client/_app/immutable/nodes/{10.D1wl2wPX.js → 10.CF7RGXpe.js} +1 -1
  29. package/build/client/_app/immutable/nodes/10.CF7RGXpe.js.br +0 -0
  30. package/build/client/_app/immutable/nodes/10.CF7RGXpe.js.gz +0 -0
  31. package/build/client/_app/immutable/nodes/11.BV_G7yLI.js +2 -0
  32. package/build/client/_app/immutable/nodes/11.BV_G7yLI.js.br +0 -0
  33. package/build/client/_app/immutable/nodes/11.BV_G7yLI.js.gz +0 -0
  34. package/build/client/_app/immutable/nodes/{2.D1Mm0DUX.js → 2.DcRhsjYp.js} +1 -1
  35. package/build/client/_app/immutable/nodes/2.DcRhsjYp.js.br +0 -0
  36. package/build/client/_app/immutable/nodes/2.DcRhsjYp.js.gz +0 -0
  37. package/build/client/_app/immutable/nodes/{3.Wfz3TcJd.js → 3.0MMe3oxR.js} +1 -1
  38. package/build/client/_app/immutable/nodes/3.0MMe3oxR.js.br +0 -0
  39. package/build/client/_app/immutable/nodes/3.0MMe3oxR.js.gz +0 -0
  40. package/build/client/_app/immutable/nodes/{6.DtZAEPXb.js → 6.ComiWlV6.js} +1 -1
  41. package/build/client/_app/immutable/nodes/6.ComiWlV6.js.br +0 -0
  42. package/build/client/_app/immutable/nodes/6.ComiWlV6.js.gz +0 -0
  43. package/build/client/_app/immutable/nodes/{7.MfBRh32I.js → 7.vkPx1kVP.js} +1 -1
  44. package/build/client/_app/immutable/nodes/7.vkPx1kVP.js.br +0 -0
  45. package/build/client/_app/immutable/nodes/7.vkPx1kVP.js.gz +0 -0
  46. package/build/client/_app/immutable/nodes/{8.DVE6LnOC.js → 8.Bmr3sWbS.js} +1 -1
  47. package/build/client/_app/immutable/nodes/8.Bmr3sWbS.js.br +0 -0
  48. package/build/client/_app/immutable/nodes/8.Bmr3sWbS.js.gz +0 -0
  49. package/build/client/_app/immutable/nodes/{9.BCel5OqI.js → 9.CAJucyeI.js} +1 -1
  50. package/build/client/_app/immutable/nodes/9.CAJucyeI.js.br +0 -0
  51. package/build/client/_app/immutable/nodes/9.CAJucyeI.js.gz +0 -0
  52. package/build/client/_app/version.json +1 -1
  53. package/build/client/_app/version.json.br +0 -0
  54. package/build/client/_app/version.json.gz +0 -0
  55. package/build/server/chunks/{0-DJqyZZTr.js → 0-DDGB6CRT.js} +2 -2
  56. package/build/server/chunks/{0-DJqyZZTr.js.map → 0-DDGB6CRT.js.map} +1 -1
  57. package/build/server/chunks/{1-2YUVen1F.js → 1-DEjonQXD.js} +2 -2
  58. package/build/server/chunks/{1-2YUVen1F.js.map → 1-DEjonQXD.js.map} +1 -1
  59. package/build/server/chunks/{10-D1X7LB3v.js → 10-BK1kiiiw.js} +2 -2
  60. package/build/server/chunks/{10-D1X7LB3v.js.map → 10-BK1kiiiw.js.map} +1 -1
  61. package/build/server/chunks/{11-qXSPdF5j.js → 11-CJPjkEF3.js} +4 -4
  62. package/build/server/chunks/11-CJPjkEF3.js.map +1 -0
  63. package/build/server/chunks/{2-BD7kj1mt.js → 2-RLnhlWh5.js} +2 -2
  64. package/build/server/chunks/{2-BD7kj1mt.js.map → 2-RLnhlWh5.js.map} +1 -1
  65. package/build/server/chunks/{3-oNjv-BhZ.js → 3-Dd4pJBqZ.js} +2 -2
  66. package/build/server/chunks/{3-oNjv-BhZ.js.map → 3-Dd4pJBqZ.js.map} +1 -1
  67. package/build/server/chunks/{6-DRJGUqHG.js → 6-DdRMnKNa.js} +2 -2
  68. package/build/server/chunks/{6-DRJGUqHG.js.map → 6-DdRMnKNa.js.map} +1 -1
  69. package/build/server/chunks/{7-_giJiu0L.js → 7-vLOMMetm.js} +2 -2
  70. package/build/server/chunks/{7-_giJiu0L.js.map → 7-vLOMMetm.js.map} +1 -1
  71. package/build/server/chunks/{8-zvWAVNT5.js → 8-rJyiQLFs.js} +2 -2
  72. package/build/server/chunks/{8-zvWAVNT5.js.map → 8-rJyiQLFs.js.map} +1 -1
  73. package/build/server/chunks/{9-DVyDL445.js → 9-CVSNNYED.js} +2 -2
  74. package/build/server/chunks/{9-DVyDL445.js.map → 9-CVSNNYED.js.map} +1 -1
  75. package/build/server/chunks/Banner-BgaAs1rs.js.map +1 -1
  76. package/build/server/chunks/Button-D0hZ7JYt.js.map +1 -1
  77. package/build/server/chunks/Icon-D0GBnDcs.js.map +1 -1
  78. package/build/server/chunks/Input-OmIiydSx.js.map +1 -1
  79. package/build/server/chunks/Pill-4xJ-VhAA.js.map +1 -1
  80. package/build/server/chunks/Shimmer-Dw2uvTC1.js.map +1 -1
  81. package/build/server/chunks/_error.svelte-CZnkxeLr.js.map +1 -1
  82. package/build/server/chunks/_page.svelte-BLo2v_8E.js.map +1 -1
  83. package/build/server/chunks/_page.svelte-BTlfUsBp.js.map +1 -1
  84. package/build/server/chunks/_page.svelte-BX2FMgSg.js.map +1 -1
  85. package/build/server/chunks/_page.svelte-C7B0qdrC.js.map +1 -1
  86. package/build/server/chunks/_page.svelte-CE7COWnF.js.map +1 -1
  87. package/build/server/chunks/_page.svelte-CWsjjd4l.js.map +1 -1
  88. package/build/server/chunks/_page.svelte-D5S2hkBk.js.map +1 -1
  89. package/build/server/chunks/_page.svelte-D_Ey8QRG.js.map +1 -1
  90. package/build/server/chunks/{_page.svelte-BUBLUSGo.js → _page.svelte-dabsQl9c.js} +206 -5
  91. package/build/server/chunks/_page.svelte-dabsQl9c.js.map +1 -0
  92. package/build/server/chunks/_page.svelte-tBuIq8Pg.js.map +1 -1
  93. package/build/server/chunks/{_server.ts-C_OOUqsd.js → _server.ts-AnBXfZXh.js} +2 -2
  94. package/build/server/chunks/{_server.ts-C_OOUqsd.js.map → _server.ts-AnBXfZXh.js.map} +1 -1
  95. package/build/server/chunks/_server.ts-B-evHL2q.js +13 -0
  96. package/build/server/chunks/_server.ts-B-evHL2q.js.map +1 -0
  97. package/build/server/chunks/_server.ts-B2wIgsW4.js +95 -0
  98. package/build/server/chunks/_server.ts-B2wIgsW4.js.map +1 -0
  99. package/build/server/chunks/{_server.ts-Bi0Oe4PF.js → _server.ts-CJGyN8mw.js} +14 -9
  100. package/build/server/chunks/_server.ts-CJGyN8mw.js.map +1 -0
  101. package/build/server/chunks/{_server.ts-DhJx0DLr.js → _server.ts-DEx9-epI.js} +16 -7
  102. package/build/server/chunks/_server.ts-DEx9-epI.js.map +1 -0
  103. package/build/server/chunks/{_server.ts-DxT9IlZF.js → _server.ts-DKNIsQeH.js} +3 -3
  104. package/build/server/chunks/{_server.ts-DxT9IlZF.js.map → _server.ts-DKNIsQeH.js.map} +1 -1
  105. package/build/server/chunks/_server.ts-DpRr0Tfh.js +68 -0
  106. package/build/server/chunks/_server.ts-DpRr0Tfh.js.map +1 -0
  107. package/build/server/chunks/{_server.ts-CRVNEOd2.js → _server.ts-Dz9Jd9Jh.js} +3 -3
  108. package/build/server/chunks/{_server.ts-CRVNEOd2.js.map → _server.ts-Dz9Jd9Jh.js.map} +1 -1
  109. package/build/server/chunks/{_server.ts-Bjbr7glm.js → _server.ts-QN-Bo5ql.js} +12 -5
  110. package/build/server/chunks/_server.ts-QN-Bo5ql.js.map +1 -0
  111. package/build/server/chunks/{_server.ts-BrqaMMAa.js → _server.ts-W6i3EnGX.js} +29 -6
  112. package/build/server/chunks/_server.ts-W6i3EnGX.js.map +1 -0
  113. package/build/server/chunks/{_server.ts-DMm0hBP4.js → _server.ts-bk_EeAdY.js} +2 -2
  114. package/build/server/chunks/{_server.ts-DMm0hBP4.js.map → _server.ts-bk_EeAdY.js.map} +1 -1
  115. package/build/server/chunks/cache-BlMaDsHi.js.map +1 -1
  116. package/build/server/chunks/guest-registry-t0-7Zv5q.js +39 -0
  117. package/build/server/chunks/guest-registry-t0-7Zv5q.js.map +1 -0
  118. package/build/server/chunks/index-CoYB03g7.js.map +1 -1
  119. package/build/server/chunks/index2-dSGQ9Eaa.js.map +1 -1
  120. package/build/server/chunks/{pty-manager-41h3IK8K.js → pty-manager-CkZNoW1t.js} +6 -2
  121. package/build/server/chunks/pty-manager-CkZNoW1t.js.map +1 -0
  122. package/build/server/chunks/root-D4IoFC8F.js.map +1 -1
  123. package/build/server/chunks/share-auth-BS7JuiHf.js +27 -0
  124. package/build/server/chunks/share-auth-BS7JuiHf.js.map +1 -0
  125. package/build/server/chunks/share-store-B9jMpVg0.js +127 -0
  126. package/build/server/chunks/share-store-B9jMpVg0.js.map +1 -0
  127. package/build/server/chunks/state.svelte-CmHqngc_.js.map +1 -1
  128. package/build/server/chunks/stores-CRYxfF0o.js.map +1 -1
  129. package/build/server/index.js +1 -1
  130. package/build/server/index.js.map +1 -1
  131. package/build/server/manifest.js +40 -19
  132. package/build/server/manifest.js.map +1 -1
  133. package/package.json +2 -2
  134. package/server.ts +8 -3
  135. package/src/lib/modules/client/terminal/ShareGate.svelte +96 -0
  136. package/src/lib/modules/client/terminal/ShareSheet.svelte +395 -0
  137. package/src/lib/modules/client/terminal/xterm-wrapper.ts +19 -2
  138. package/src/lib/modules/server/terminal/pty-manager.ts +6 -0
  139. package/src/lib/modules/server/terminal/share-auth.ts +37 -0
  140. package/src/lib/modules/server/terminal/share-store.ts +172 -0
  141. package/src/lib/modules/server/ws/guest-registry.ts +49 -0
  142. package/src/lib/modules/server/ws/server.ts +22 -3
  143. package/src/lib/modules/server/ws/session-handler.ts +18 -4
  144. package/src/lib/modules/server/ws/terminal-handler.ts +21 -2
  145. package/src/lib/modules/server/ws/ticket-store.ts +18 -10
  146. package/src/lib/types/generated/Client.ts +25 -1
  147. package/src/lib/types/generated/Share.ts +404 -0
  148. package/src/lib/types/generated/WsProtocol.ts +73 -2
  149. package/src/lib/types/generated/index.ts +1 -0
  150. package/src/lib/types/terminal-client.ts +19 -2
  151. package/src/lib/types/ws.ts +1 -0
  152. package/src/routes/api/terminals/[id]/+server.ts +14 -3
  153. package/src/routes/api/terminals/[id]/paste-image/+server.ts +8 -4
  154. package/src/routes/api/terminals/[id]/resize/+server.ts +8 -4
  155. package/src/routes/api/terminals/[id]/share/+server.ts +98 -0
  156. package/src/routes/api/terminals/[id]/share/auth/+server.ts +81 -0
  157. package/src/routes/api/terminals/[id]/share/status/+server.ts +11 -0
  158. package/src/routes/api/ws-ticket/+server.ts +26 -5
  159. package/src/routes/terminals/[id]/+page.svelte +184 -43
  160. package/build/client/_app/immutable/assets/11.v5KA95xm.css +0 -1
  161. package/build/client/_app/immutable/assets/11.v5KA95xm.css.br +0 -0
  162. package/build/client/_app/immutable/assets/11.v5KA95xm.css.gz +0 -0
  163. package/build/client/_app/immutable/chunks/BcqA7eKM.js +0 -3
  164. package/build/client/_app/immutable/chunks/BcqA7eKM.js.br +0 -0
  165. package/build/client/_app/immutable/chunks/BcqA7eKM.js.gz +0 -0
  166. package/build/client/_app/immutable/chunks/CR6bkGJW.js +0 -6
  167. package/build/client/_app/immutable/chunks/CR6bkGJW.js.br +0 -0
  168. package/build/client/_app/immutable/chunks/CR6bkGJW.js.gz +0 -0
  169. package/build/client/_app/immutable/chunks/DA4Zt9Me.js.br +0 -0
  170. package/build/client/_app/immutable/chunks/DCDL_9ys.js.br +0 -0
  171. package/build/client/_app/immutable/chunks/DCDL_9ys.js.gz +0 -0
  172. package/build/client/_app/immutable/entry/app.D4TXlu7A.js.br +0 -0
  173. package/build/client/_app/immutable/entry/app.D4TXlu7A.js.gz +0 -0
  174. package/build/client/_app/immutable/entry/start.BBQhtURO.js +0 -1
  175. package/build/client/_app/immutable/entry/start.BBQhtURO.js.br +0 -0
  176. package/build/client/_app/immutable/entry/start.BBQhtURO.js.gz +0 -0
  177. package/build/client/_app/immutable/nodes/0.1zylwAPT.js.br +0 -0
  178. package/build/client/_app/immutable/nodes/0.1zylwAPT.js.gz +0 -0
  179. package/build/client/_app/immutable/nodes/1.BVnLUSs-.js.br +0 -0
  180. package/build/client/_app/immutable/nodes/1.BVnLUSs-.js.gz +0 -0
  181. package/build/client/_app/immutable/nodes/10.D1wl2wPX.js.br +0 -0
  182. package/build/client/_app/immutable/nodes/10.D1wl2wPX.js.gz +0 -0
  183. package/build/client/_app/immutable/nodes/11.C18nMGmp.js +0 -2
  184. package/build/client/_app/immutable/nodes/11.C18nMGmp.js.br +0 -0
  185. package/build/client/_app/immutable/nodes/11.C18nMGmp.js.gz +0 -0
  186. package/build/client/_app/immutable/nodes/2.D1Mm0DUX.js.br +0 -0
  187. package/build/client/_app/immutable/nodes/2.D1Mm0DUX.js.gz +0 -0
  188. package/build/client/_app/immutable/nodes/3.Wfz3TcJd.js.br +0 -0
  189. package/build/client/_app/immutable/nodes/3.Wfz3TcJd.js.gz +0 -0
  190. package/build/client/_app/immutable/nodes/6.DtZAEPXb.js.br +0 -0
  191. package/build/client/_app/immutable/nodes/6.DtZAEPXb.js.gz +0 -0
  192. package/build/client/_app/immutable/nodes/7.MfBRh32I.js.br +0 -0
  193. package/build/client/_app/immutable/nodes/7.MfBRh32I.js.gz +0 -0
  194. package/build/client/_app/immutable/nodes/8.DVE6LnOC.js.br +0 -0
  195. package/build/client/_app/immutable/nodes/8.DVE6LnOC.js.gz +0 -0
  196. package/build/client/_app/immutable/nodes/9.BCel5OqI.js.br +0 -0
  197. package/build/client/_app/immutable/nodes/9.BCel5OqI.js.gz +0 -0
  198. package/build/server/chunks/11-qXSPdF5j.js.map +0 -1
  199. package/build/server/chunks/_page.svelte-BUBLUSGo.js.map +0 -1
  200. package/build/server/chunks/_server.ts-Bi0Oe4PF.js.map +0 -1
  201. package/build/server/chunks/_server.ts-Bjbr7glm.js.map +0 -1
  202. package/build/server/chunks/_server.ts-BrqaMMAa.js.map +0 -1
  203. package/build/server/chunks/_server.ts-DhJx0DLr.js.map +0 -1
  204. package/build/server/chunks/events-handler-Dm1mNPQP.js +0 -20
  205. package/build/server/chunks/events-handler-Dm1mNPQP.js.map +0 -1
  206. package/build/server/chunks/pty-manager-41h3IK8K.js.map +0 -1
@@ -0,0 +1,404 @@
1
+ import {
2
+ isJSON,
3
+ decodeString,
4
+ _decodeString,
5
+ decodeNumber,
6
+ _decodeNumber,
7
+ decodeBoolean,
8
+ _decodeBoolean,
9
+ } from 'type-decoder';
10
+
11
+ /**
12
+ * @type { ShareMode }
13
+ * @description What a share guest may do — watch only, or full input control
14
+ */
15
+ export type ShareMode = 'view' | 'control';
16
+
17
+ export function decodeShareMode(rawInput: unknown): ShareMode | null {
18
+ switch (rawInput) {
19
+ case 'view':
20
+ case 'control':
21
+ return rawInput;
22
+ }
23
+ return null;
24
+ }
25
+
26
+ export function _decodeShareMode(rawInput: unknown): ShareMode | undefined {
27
+ switch (rawInput) {
28
+ case 'view':
29
+ case 'control':
30
+ return rawInput;
31
+ }
32
+ return;
33
+ }
34
+
35
+ /**
36
+ * @type { AccessLevel }
37
+ * @description Authorization level resolved for a terminal-scoped request
38
+ */
39
+ export type AccessLevel = 'owner' | 'guest';
40
+
41
+ export function decodeAccessLevel(rawInput: unknown): AccessLevel | null {
42
+ switch (rawInput) {
43
+ case 'owner':
44
+ case 'guest':
45
+ return rawInput;
46
+ }
47
+ return null;
48
+ }
49
+
50
+ export function _decodeAccessLevel(rawInput: unknown): AccessLevel | undefined {
51
+ switch (rawInput) {
52
+ case 'owner':
53
+ case 'guest':
54
+ return rawInput;
55
+ }
56
+ return;
57
+ }
58
+
59
+ /**
60
+ * @type { AccessContext }
61
+ * @description Resolved authorization for a terminal-scoped API request
62
+ */
63
+ export type AccessContext = {
64
+ /**
65
+ * @description owner (API key) or guest (share session token)
66
+ * @type { AccessLevel }
67
+ * @memberof AccessContext
68
+ */
69
+ level: AccessLevel;
70
+ /**
71
+ * @description Guest access mode (present only when level is guest)
72
+ * @type { ShareMode }
73
+ * @memberof AccessContext
74
+ */
75
+ mode: ShareMode | null;
76
+ };
77
+
78
+ export function decodeAccessContext(rawInput: unknown): AccessContext | null {
79
+ if (isJSON(rawInput)) {
80
+ const decodedLevel = decodeAccessLevel(rawInput['level']);
81
+ const decodedMode = decodeShareMode(rawInput['mode']);
82
+
83
+ if (decodedLevel === null) {
84
+ return null;
85
+ }
86
+
87
+ return {
88
+ level: decodedLevel,
89
+ mode: decodedMode,
90
+ };
91
+ }
92
+ return null;
93
+ }
94
+
95
+ /**
96
+ * @type { TerminalShareRecord }
97
+ * @description Persisted share configuration for one terminal (terminal_shares table)
98
+ */
99
+ export type TerminalShareRecord = {
100
+ /**
101
+ * @description Terminal this share belongs to (primary key)
102
+ * @type { string }
103
+ * @memberof TerminalShareRecord
104
+ */
105
+ terminalId: string;
106
+ /**
107
+ * @description scrypt:<salt-hex>:<hash-hex> password hash
108
+ * @type { string }
109
+ * @memberof TerminalShareRecord
110
+ */
111
+ passwordHash: string;
112
+ /**
113
+ * @description Access mode granted to guests
114
+ * @type { ShareMode }
115
+ * @memberof TerminalShareRecord
116
+ */
117
+ mode: ShareMode;
118
+ /**
119
+ * @description Unix timestamp (ms) when the share was created
120
+ * @type { number }
121
+ * @memberof TerminalShareRecord
122
+ */
123
+ createdAt: number;
124
+ /**
125
+ * @description Unix timestamp (ms) when the share was last updated
126
+ * @type { number }
127
+ * @memberof TerminalShareRecord
128
+ */
129
+ updatedAt: number;
130
+ };
131
+
132
+ export function decodeTerminalShareRecord(rawInput: unknown): TerminalShareRecord | null {
133
+ if (isJSON(rawInput)) {
134
+ const decodedTerminalId = decodeString(rawInput['terminalId']);
135
+ const decodedPasswordHash = decodeString(rawInput['passwordHash']);
136
+ const decodedMode = decodeShareMode(rawInput['mode']);
137
+ const decodedCreatedAt = decodeNumber(rawInput['createdAt']);
138
+ const decodedUpdatedAt = decodeNumber(rawInput['updatedAt']);
139
+
140
+ if (
141
+ decodedTerminalId === null ||
142
+ decodedPasswordHash === null ||
143
+ decodedMode === null ||
144
+ decodedCreatedAt === null ||
145
+ decodedUpdatedAt === null
146
+ ) {
147
+ return null;
148
+ }
149
+
150
+ return {
151
+ terminalId: decodedTerminalId,
152
+ passwordHash: decodedPasswordHash,
153
+ mode: decodedMode,
154
+ createdAt: decodedCreatedAt,
155
+ updatedAt: decodedUpdatedAt,
156
+ };
157
+ }
158
+ return null;
159
+ }
160
+
161
+ /**
162
+ * @type { ShareSessionRecord }
163
+ * @description Persisted guest session (share_sessions table); token stored as sha256 hash
164
+ */
165
+ export type ShareSessionRecord = {
166
+ /**
167
+ * @description sha256 hex of the guest bearer token (primary key)
168
+ * @type { string }
169
+ * @memberof ShareSessionRecord
170
+ */
171
+ tokenHash: string;
172
+ /**
173
+ * @description Terminal the session grants access to
174
+ * @type { string }
175
+ * @memberof ShareSessionRecord
176
+ */
177
+ terminalId: string;
178
+ /**
179
+ * @description Unix timestamp (ms) when the session was created
180
+ * @type { number }
181
+ * @memberof ShareSessionRecord
182
+ */
183
+ createdAt: number;
184
+ /**
185
+ * @description Unix timestamp (ms) when the session expires (created + 7 days)
186
+ * @type { number }
187
+ * @memberof ShareSessionRecord
188
+ */
189
+ expiresAt: number;
190
+ };
191
+
192
+ export function decodeShareSessionRecord(rawInput: unknown): ShareSessionRecord | null {
193
+ if (isJSON(rawInput)) {
194
+ const decodedTokenHash = decodeString(rawInput['tokenHash']);
195
+ const decodedTerminalId = decodeString(rawInput['terminalId']);
196
+ const decodedCreatedAt = decodeNumber(rawInput['createdAt']);
197
+ const decodedExpiresAt = decodeNumber(rawInput['expiresAt']);
198
+
199
+ if (
200
+ decodedTokenHash === null ||
201
+ decodedTerminalId === null ||
202
+ decodedCreatedAt === null ||
203
+ decodedExpiresAt === null
204
+ ) {
205
+ return null;
206
+ }
207
+
208
+ return {
209
+ tokenHash: decodedTokenHash,
210
+ terminalId: decodedTerminalId,
211
+ createdAt: decodedCreatedAt,
212
+ expiresAt: decodedExpiresAt,
213
+ };
214
+ }
215
+ return null;
216
+ }
217
+
218
+ /**
219
+ * @type { ShareInfoResponse }
220
+ * @description Owner view of a terminal's share state (GET /api/terminals/[id]/share)
221
+ */
222
+ export type ShareInfoResponse = {
223
+ /**
224
+ * @description Whether sharing is currently enabled for this terminal
225
+ * @type { boolean }
226
+ * @memberof ShareInfoResponse
227
+ */
228
+ active: boolean;
229
+ /**
230
+ * @description Current access mode (present when active)
231
+ * @type { ShareMode }
232
+ * @memberof ShareInfoResponse
233
+ */
234
+ mode: ShareMode | null;
235
+ /**
236
+ * @description Unix timestamp (ms) when the share was created (present when active)
237
+ * @type { number }
238
+ * @memberof ShareInfoResponse
239
+ */
240
+ createdAt: number | null;
241
+ /**
242
+ * @description Unix timestamp (ms) when the share was last updated (present when active)
243
+ * @type { number }
244
+ * @memberof ShareInfoResponse
245
+ */
246
+ updatedAt: number | null;
247
+ };
248
+
249
+ export function decodeShareInfoResponse(rawInput: unknown): ShareInfoResponse | null {
250
+ if (isJSON(rawInput)) {
251
+ const decodedActive = decodeBoolean(rawInput['active']);
252
+ const decodedMode = decodeShareMode(rawInput['mode']);
253
+ const decodedCreatedAt = decodeNumber(rawInput['createdAt']);
254
+ const decodedUpdatedAt = decodeNumber(rawInput['updatedAt']);
255
+
256
+ if (decodedActive === null) {
257
+ return null;
258
+ }
259
+
260
+ return {
261
+ active: decodedActive,
262
+ mode: decodedMode,
263
+ createdAt: decodedCreatedAt,
264
+ updatedAt: decodedUpdatedAt,
265
+ };
266
+ }
267
+ return null;
268
+ }
269
+
270
+ /**
271
+ * @type { ShareStatusResponse }
272
+ * @description Public probe response (GET /api/terminals/[id]/share/status)
273
+ */
274
+ export type ShareStatusResponse = {
275
+ /**
276
+ * @description Whether a share exists for this terminal
277
+ * @type { boolean }
278
+ * @memberof ShareStatusResponse
279
+ */
280
+ shared: boolean;
281
+ };
282
+
283
+ export function decodeShareStatusResponse(rawInput: unknown): ShareStatusResponse | null {
284
+ if (isJSON(rawInput)) {
285
+ const decodedShared = decodeBoolean(rawInput['shared']);
286
+
287
+ if (decodedShared === null) {
288
+ return null;
289
+ }
290
+
291
+ return {
292
+ shared: decodedShared,
293
+ };
294
+ }
295
+ return null;
296
+ }
297
+
298
+ /**
299
+ * @type { ShareAuthRequest }
300
+ * @description Guest password exchange request (POST /api/terminals/[id]/share/auth)
301
+ */
302
+ export type ShareAuthRequest = {
303
+ /**
304
+ * @description The share password
305
+ * @type { string }
306
+ * @memberof ShareAuthRequest
307
+ */
308
+ password: string;
309
+ };
310
+
311
+ export function decodeShareAuthRequest(rawInput: unknown): ShareAuthRequest | null {
312
+ if (isJSON(rawInput)) {
313
+ const decodedPassword = decodeString(rawInput['password']);
314
+
315
+ if (decodedPassword === null) {
316
+ return null;
317
+ }
318
+
319
+ return {
320
+ password: decodedPassword,
321
+ };
322
+ }
323
+ return null;
324
+ }
325
+
326
+ /**
327
+ * @type { ShareAuthResponse }
328
+ * @description Guest session issued after successful password exchange
329
+ */
330
+ export type ShareAuthResponse = {
331
+ /**
332
+ * @description Guest bearer token (64-char hex, shown once)
333
+ * @type { string }
334
+ * @memberof ShareAuthResponse
335
+ */
336
+ token: string;
337
+ /**
338
+ * @description Access mode granted by this session
339
+ * @type { ShareMode }
340
+ * @memberof ShareAuthResponse
341
+ */
342
+ mode: ShareMode;
343
+ /**
344
+ * @description Unix timestamp (ms) when this session expires
345
+ * @type { number }
346
+ * @memberof ShareAuthResponse
347
+ */
348
+ expiresAt: number;
349
+ };
350
+
351
+ export function decodeShareAuthResponse(rawInput: unknown): ShareAuthResponse | null {
352
+ if (isJSON(rawInput)) {
353
+ const decodedToken = decodeString(rawInput['token']);
354
+ const decodedMode = decodeShareMode(rawInput['mode']);
355
+ const decodedExpiresAt = decodeNumber(rawInput['expiresAt']);
356
+
357
+ if (decodedToken === null || decodedMode === null || decodedExpiresAt === null) {
358
+ return null;
359
+ }
360
+
361
+ return {
362
+ token: decodedToken,
363
+ mode: decodedMode,
364
+ expiresAt: decodedExpiresAt,
365
+ };
366
+ }
367
+ return null;
368
+ }
369
+
370
+ /**
371
+ * @type { ShareConfigRequest }
372
+ * @description Owner create/update share request (PUT /api/terminals/[id]/share)
373
+ */
374
+ export type ShareConfigRequest = {
375
+ /**
376
+ * @description New share password (min 6 chars; required on create, optional on update)
377
+ * @type { string }
378
+ * @memberof ShareConfigRequest
379
+ */
380
+ password: string | null;
381
+ /**
382
+ * @description Access mode to grant guests
383
+ * @type { ShareMode }
384
+ * @memberof ShareConfigRequest
385
+ */
386
+ mode: ShareMode;
387
+ };
388
+
389
+ export function decodeShareConfigRequest(rawInput: unknown): ShareConfigRequest | null {
390
+ if (isJSON(rawInput)) {
391
+ const decodedPassword = decodeString(rawInput['password']);
392
+ const decodedMode = decodeShareMode(rawInput['mode']);
393
+
394
+ if (decodedMode === null) {
395
+ return null;
396
+ }
397
+
398
+ return {
399
+ password: decodedPassword,
400
+ mode: decodedMode,
401
+ };
402
+ }
403
+ return null;
404
+ }
@@ -435,7 +435,8 @@ export type TerminalServerMessage =
435
435
  | CTerminalServerMessageTerminalOutputDroppedMessage
436
436
  | CTerminalServerMessageTerminalScrollbackMessage
437
437
  | CTerminalServerMessageTerminalExitMessage
438
- | CTerminalServerMessageTerminalErrorMessage;
438
+ | CTerminalServerMessageTerminalErrorMessage
439
+ | CTerminalServerMessageTerminalResizeMessage;
439
440
 
440
441
  export function decodeTerminalServerMessage(rawInput: unknown): TerminalServerMessage | null {
441
442
  const result: TerminalServerMessage | null =
@@ -443,7 +444,8 @@ export function decodeTerminalServerMessage(rawInput: unknown): TerminalServerMe
443
444
  decodeCTerminalServerMessageTerminalOutputDroppedMessage(rawInput) ??
444
445
  decodeCTerminalServerMessageTerminalScrollbackMessage(rawInput) ??
445
446
  decodeCTerminalServerMessageTerminalExitMessage(rawInput) ??
446
- decodeCTerminalServerMessageTerminalErrorMessage(rawInput);
447
+ decodeCTerminalServerMessageTerminalErrorMessage(rawInput) ??
448
+ decodeCTerminalServerMessageTerminalResizeMessage(rawInput);
447
449
 
448
450
  return result;
449
451
  }
@@ -533,6 +535,23 @@ export function decodeCTerminalServerMessageTerminalErrorMessage(
533
535
  return new CTerminalServerMessageTerminalErrorMessage(result);
534
536
  }
535
537
 
538
+ export class CTerminalServerMessageTerminalResizeMessage {
539
+ data: TerminalResizeMessage;
540
+ constructor(data: TerminalResizeMessage) {
541
+ this.data = data;
542
+ }
543
+ }
544
+
545
+ export function decodeCTerminalServerMessageTerminalResizeMessage(
546
+ rawInput: unknown
547
+ ): CTerminalServerMessageTerminalResizeMessage | null {
548
+ const result = decodeTerminalResizeMessage(rawInput);
549
+ if (result === null) {
550
+ return null;
551
+ }
552
+ return new CTerminalServerMessageTerminalResizeMessage(result);
553
+ }
554
+
536
555
  /**
537
556
  * @type { TextContentBlock }
538
557
  * @description Content block in the live 'message' payload
@@ -2326,12 +2345,26 @@ export type Ticket = {
2326
2345
  * @memberof Ticket
2327
2346
  */
2328
2347
  used: boolean;
2348
+ /**
2349
+ * @description When set, the ticket may only open WS channels for this terminal
2350
+ * @type { string }
2351
+ * @memberof Ticket
2352
+ */
2353
+ terminalId: string | null;
2354
+ /**
2355
+ * @description When true, input/resize/signal/send-input/cancel frames are dropped
2356
+ * @type { boolean }
2357
+ * @memberof Ticket
2358
+ */
2359
+ readOnly: boolean | null;
2329
2360
  };
2330
2361
 
2331
2362
  export function decodeTicket(rawInput: unknown): Ticket | null {
2332
2363
  if (isJSON(rawInput)) {
2333
2364
  const decodedCreatedAt = decodeNumber(rawInput['createdAt']);
2334
2365
  const decodedUsed = decodeBoolean(rawInput['used']);
2366
+ const decodedTerminalId = decodeString(rawInput['terminalId']);
2367
+ const decodedReadOnly = decodeBoolean(rawInput['readOnly']);
2335
2368
 
2336
2369
  if (decodedCreatedAt === null || decodedUsed === null) {
2337
2370
  return null;
@@ -2340,6 +2373,44 @@ export function decodeTicket(rawInput: unknown): Ticket | null {
2340
2373
  return {
2341
2374
  createdAt: decodedCreatedAt,
2342
2375
  used: decodedUsed,
2376
+ terminalId: decodedTerminalId,
2377
+ readOnly: decodedReadOnly,
2378
+ };
2379
+ }
2380
+ return null;
2381
+ }
2382
+
2383
+ /**
2384
+ * @type { TicketScope }
2385
+ * @description Scope restriction carried by a guest WebSocket ticket
2386
+ */
2387
+ export type TicketScope = {
2388
+ /**
2389
+ * @description Only WS channels for this terminal may be opened
2390
+ * @type { string }
2391
+ * @memberof TicketScope
2392
+ */
2393
+ terminalId: string;
2394
+ /**
2395
+ * @description Whether the connection is view-only (input frames dropped)
2396
+ * @type { boolean }
2397
+ * @memberof TicketScope
2398
+ */
2399
+ readOnly: boolean;
2400
+ };
2401
+
2402
+ export function decodeTicketScope(rawInput: unknown): TicketScope | null {
2403
+ if (isJSON(rawInput)) {
2404
+ const decodedTerminalId = decodeString(rawInput['terminalId']);
2405
+ const decodedReadOnly = decodeBoolean(rawInput['readOnly']);
2406
+
2407
+ if (decodedTerminalId === null || decodedReadOnly === null) {
2408
+ return null;
2409
+ }
2410
+
2411
+ return {
2412
+ terminalId: decodedTerminalId,
2413
+ readOnly: decodedReadOnly,
2343
2414
  };
2344
2415
  }
2345
2416
  return null;
@@ -10,3 +10,4 @@ export * from './OpenCode';
10
10
  export * from './Notification';
11
11
  export * from './Client';
12
12
  export * from './Config';
13
+ export * from './Share';
@@ -80,12 +80,24 @@ export interface QuickKeysProps {
80
80
 
81
81
  // --- keyboard-shortcuts types ---
82
82
 
83
- export interface ShortcutManagerOptions {
84
- onHelp: () => void;
83
+ export interface ShareGateProps {
84
+ /** Returns an error message to display, or null on success. */
85
+ onSubmit: (password: string) => Promise<null | string>;
85
86
  }
86
87
 
87
88
  // --- xterm-wrapper types ---
88
89
 
90
+ export interface ShareSheetProps {
91
+ onClose: () => void;
92
+ open?: boolean;
93
+ shareUrl: string;
94
+ terminalId: string;
95
+ }
96
+
97
+ export interface ShortcutManagerOptions {
98
+ onHelp: () => void;
99
+ }
100
+
89
101
  export interface ShortcutsHelpProps {
90
102
  onClose: () => void;
91
103
  open: boolean;
@@ -105,11 +117,14 @@ export interface TerminalOptions {
105
117
  container: HTMLElement;
106
118
  fontSize?: number;
107
119
  getTicket: () => Promise<string>;
120
+ initialCols?: number;
121
+ initialRows?: number;
108
122
  onActivity?: (active: boolean) => void;
109
123
  onCwd?: (path: string) => void;
110
124
  onDisconnect?: () => void;
111
125
  onExit?: (code: number) => void;
112
126
  onReconnect?: () => void;
127
+ readOnly?: boolean;
113
128
  terminalId?: string;
114
129
  wsUrl: string;
115
130
  }
@@ -126,7 +141,9 @@ export interface WsTerminalInboundMessage {
126
141
  active?: boolean;
127
142
  bytes?: number;
128
143
  code?: number;
144
+ cols?: number;
129
145
  data?: string;
130
146
  path?: string;
147
+ rows?: number;
131
148
  type: string;
132
149
  }
@@ -146,6 +146,7 @@ export type WireTerminalServerMessage =
146
146
  | { bytes: number; type: 'output-dropped' }
147
147
  | { chunk: number; data: string; total: number; type: 'scrollback' }
148
148
  | { code: null | number; signal: null | string; type: 'exit' }
149
+ | { cols: number; rows: number; type: 'resize' }
149
150
  | { data: string; type: 'output' }
150
151
  | { message: string; type: 'error' };
151
152
 
@@ -1,6 +1,9 @@
1
1
  import { validateAuth } from '$lib/modules/server/auth';
2
2
  import { ptyManager } from '$lib/modules/server/terminal/pty-manager.js';
3
+ import { resolveAccess } from '$lib/modules/server/terminal/share-auth';
4
+ import { shareStore } from '$lib/modules/server/terminal/share-store';
3
5
  import { toErrorMessage } from '$lib/modules/server/utils/error';
6
+ import { closeGuests } from '$lib/modules/server/ws/guest-registry';
4
7
  import { json } from '@sveltejs/kit';
5
8
 
6
9
  import type { RequestHandler } from './$types';
@@ -21,10 +24,11 @@ function lastScrollbackLine(scrollback: string): null | string {
21
24
  }
22
25
 
23
26
  // GET /api/terminals/:id — Get terminal details by ID
27
+ // Accepts the API key (owner) or a share session token scoped to this terminal.
24
28
  export const GET: RequestHandler = ({ params, request }) => {
25
- const authError = validateAuth(request);
26
- if (authError) {
27
- return authError;
29
+ const access = resolveAccess(request, params.id);
30
+ if (!access) {
31
+ return json({ error: 'Unauthorized' }, { status: 401 });
28
32
  }
29
33
 
30
34
  try {
@@ -37,6 +41,7 @@ export const GET: RequestHandler = ({ params, request }) => {
37
41
  return json({
38
42
  args: terminal.args,
39
43
  clientCount: terminal.clients.size,
44
+ cols: terminal.cols,
40
45
  command: terminal.command,
41
46
  createdAt: terminal.createdAt.toISOString(),
42
47
  cwd: terminal.cwd,
@@ -45,10 +50,12 @@ export const GET: RequestHandler = ({ params, request }) => {
45
50
  id: terminal.id,
46
51
  lastOutput: lastScrollbackLine(terminal.scrollback),
47
52
  pid: terminal.pid,
53
+ rows: terminal.rows,
48
54
  sessionWs: `/ws/session/${terminal.id}`,
49
55
  status: terminal.status,
50
56
  timestamp: new Date().toISOString(),
51
57
  ws: `/ws/terminal/${terminal.id}`,
58
+ ...(access.level === 'guest' ? { shareMode: access.mode } : {}),
52
59
  });
53
60
  } catch (error) {
54
61
  console.error('[terminals] Failed to get terminal:', toErrorMessage(error));
@@ -72,6 +79,8 @@ export const DELETE: RequestHandler = ({ params, request }) => {
72
79
 
73
80
  if (terminal.status === 'exited') {
74
81
  ptyManager.remove(params.id);
82
+ shareStore.deleteShare(params.id);
83
+ closeGuests(params.id);
75
84
  console.log(`[terminals] Removed exited terminal ${params.id}`);
76
85
  return json({
77
86
  removed: true,
@@ -81,6 +90,8 @@ export const DELETE: RequestHandler = ({ params, request }) => {
81
90
  }
82
91
 
83
92
  ptyManager.kill(params.id);
93
+ shareStore.deleteShare(params.id);
94
+ closeGuests(params.id);
84
95
 
85
96
  console.log(`[terminals] Killed terminal ${params.id} (pid=${terminal.pid})`);
86
97
 
@@ -1,4 +1,4 @@
1
- import { validateAuth } from '$lib/modules/server/auth';
1
+ import { resolveAccess } from '$lib/modules/server/terminal/share-auth';
2
2
  import { toErrorMessage } from '$lib/modules/server/utils/error';
3
3
  import { json } from '@sveltejs/kit';
4
4
  import { mkdirSync, writeFileSync } from 'fs';
@@ -6,10 +6,14 @@ import { mkdirSync, writeFileSync } from 'fs';
6
6
  import type { RequestHandler } from './$types';
7
7
 
8
8
  // POST /api/terminals/[id]/paste-image — Write clipboard image for a terminal
9
+ // Accepts the API key (owner) or a control-mode share token for this terminal.
9
10
  export const POST: RequestHandler = async ({ params, request }) => {
10
- const authError = validateAuth(request);
11
- if (authError) {
12
- return authError;
11
+ const access = resolveAccess(request, params.id);
12
+ if (!access) {
13
+ return json({ error: 'Unauthorized' }, { status: 401 });
14
+ }
15
+ if (access.level === 'guest' && access.mode !== 'control') {
16
+ return json({ error: 'View-only access' }, { status: 403 });
13
17
  }
14
18
 
15
19
  const terminalId = params.id;
@@ -1,15 +1,19 @@
1
- import { validateAuth } from '$lib/modules/server/auth';
2
1
  import { ptyManager } from '$lib/modules/server/terminal/pty-manager.js';
2
+ import { resolveAccess } from '$lib/modules/server/terminal/share-auth';
3
3
  import { toErrorMessage } from '$lib/modules/server/utils/error';
4
4
  import { json } from '@sveltejs/kit';
5
5
 
6
6
  import type { RequestHandler } from './$types';
7
7
 
8
8
  // POST /api/terminals/:id/resize — Resize terminal
9
+ // Accepts the API key (owner) or a control-mode share token for this terminal.
9
10
  export const POST: RequestHandler = async ({ params, request }) => {
10
- const authError = validateAuth(request);
11
- if (authError) {
12
- return authError;
11
+ const access = resolveAccess(request, params.id);
12
+ if (!access) {
13
+ return json({ error: 'Unauthorized' }, { status: 401 });
14
+ }
15
+ if (access.level === 'guest' && access.mode !== 'control') {
16
+ return json({ error: 'View-only access' }, { status: 403 });
13
17
  }
14
18
 
15
19
  let body: { cols?: number; rows?: number };