@moqtap/codec 0.2.0 → 0.2.2

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 (439) hide show
  1. package/dist/core/buffer-reader.d.ts +15 -0
  2. package/dist/core/buffer-reader.d.ts.map +1 -0
  3. package/dist/core/buffer-reader.js +98 -0
  4. package/dist/core/buffer-reader.js.map +1 -0
  5. package/dist/core/buffer-writer.d.ts +16 -0
  6. package/dist/core/buffer-writer.d.ts.map +1 -0
  7. package/dist/core/buffer-writer.js +86 -0
  8. package/dist/core/buffer-writer.js.map +1 -0
  9. package/dist/core/errors.d.ts +2 -0
  10. package/dist/core/errors.d.ts.map +1 -0
  11. package/dist/core/errors.js +2 -0
  12. package/dist/core/errors.js.map +1 -0
  13. package/dist/core/hex.d.ts +5 -0
  14. package/dist/core/hex.d.ts.map +1 -0
  15. package/dist/core/hex.js +17 -0
  16. package/dist/core/hex.js.map +1 -0
  17. package/dist/core/session-types.d.ts +99 -0
  18. package/dist/core/session-types.d.ts.map +1 -0
  19. package/dist/core/session-types.js +2 -0
  20. package/dist/core/session-types.js.map +1 -0
  21. package/dist/core/types.d.ts +235 -0
  22. package/dist/core/types.d.ts.map +1 -0
  23. package/dist/core/types.js +11 -0
  24. package/dist/core/types.js.map +1 -0
  25. package/dist/drafts/draft07/announce-fsm.d.ts +2 -0
  26. package/dist/drafts/draft07/announce-fsm.d.ts.map +1 -0
  27. package/dist/drafts/draft07/announce-fsm.js +2 -0
  28. package/dist/drafts/draft07/announce-fsm.js.map +1 -0
  29. package/dist/drafts/draft07/codec.d.ts +17 -0
  30. package/dist/drafts/draft07/codec.d.ts.map +1 -0
  31. package/dist/drafts/draft07/codec.js +722 -0
  32. package/dist/drafts/draft07/codec.js.map +1 -0
  33. package/dist/drafts/draft07/data-streams.d.ts +9 -0
  34. package/dist/drafts/draft07/data-streams.d.ts.map +1 -0
  35. package/dist/drafts/draft07/data-streams.js +228 -0
  36. package/dist/drafts/draft07/data-streams.js.map +1 -0
  37. package/dist/drafts/draft07/index.d.ts +14 -0
  38. package/dist/drafts/draft07/index.d.ts.map +1 -0
  39. package/dist/drafts/draft07/index.js +18 -0
  40. package/dist/drafts/draft07/index.js.map +1 -0
  41. package/dist/drafts/draft07/messages.d.ts +32 -0
  42. package/dist/drafts/draft07/messages.d.ts.map +1 -0
  43. package/dist/drafts/draft07/messages.js +42 -0
  44. package/dist/drafts/draft07/messages.js.map +1 -0
  45. package/dist/drafts/draft07/parameters.d.ts +4 -0
  46. package/dist/drafts/draft07/parameters.d.ts.map +1 -0
  47. package/dist/drafts/draft07/parameters.js +10 -0
  48. package/dist/drafts/draft07/parameters.js.map +1 -0
  49. package/dist/drafts/draft07/rules.d.ts +8 -0
  50. package/dist/drafts/draft07/rules.d.ts.map +1 -0
  51. package/dist/drafts/draft07/rules.js +95 -0
  52. package/dist/drafts/draft07/rules.js.map +1 -0
  53. package/dist/drafts/draft07/session-fsm.d.ts +38 -0
  54. package/dist/drafts/draft07/session-fsm.d.ts.map +1 -0
  55. package/dist/drafts/draft07/session-fsm.js +354 -0
  56. package/dist/drafts/draft07/session-fsm.js.map +1 -0
  57. package/dist/drafts/draft07/session.d.ts +4 -0
  58. package/dist/drafts/draft07/session.d.ts.map +1 -0
  59. package/dist/drafts/draft07/session.js +5 -0
  60. package/dist/drafts/draft07/session.js.map +1 -0
  61. package/dist/drafts/draft07/subscription-fsm.d.ts +2 -0
  62. package/dist/drafts/draft07/subscription-fsm.d.ts.map +1 -0
  63. package/dist/drafts/draft07/subscription-fsm.js +2 -0
  64. package/dist/drafts/draft07/subscription-fsm.js.map +1 -0
  65. package/dist/drafts/draft07/types.d.ts +61 -0
  66. package/dist/drafts/draft07/types.d.ts.map +1 -0
  67. package/dist/drafts/draft07/types.js +4 -0
  68. package/dist/drafts/draft07/types.js.map +1 -0
  69. package/dist/drafts/draft07/varint.d.ts +4 -0
  70. package/dist/drafts/draft07/varint.d.ts.map +1 -0
  71. package/dist/drafts/draft07/varint.js +22 -0
  72. package/dist/drafts/draft07/varint.js.map +1 -0
  73. package/dist/drafts/draft08/codec.d.ts +29 -0
  74. package/dist/drafts/draft08/codec.d.ts.map +1 -0
  75. package/dist/drafts/draft08/codec.js +729 -0
  76. package/dist/drafts/draft08/codec.js.map +1 -0
  77. package/dist/drafts/draft08/data-streams.d.ts +12 -0
  78. package/dist/drafts/draft08/data-streams.d.ts.map +1 -0
  79. package/dist/drafts/draft08/data-streams.js +345 -0
  80. package/dist/drafts/draft08/data-streams.js.map +1 -0
  81. package/dist/drafts/draft08/index.d.ts +9 -0
  82. package/dist/drafts/draft08/index.d.ts.map +1 -0
  83. package/dist/drafts/draft08/index.js +7 -0
  84. package/dist/drafts/draft08/index.js.map +1 -0
  85. package/dist/drafts/draft08/messages.d.ts +34 -0
  86. package/dist/drafts/draft08/messages.d.ts.map +1 -0
  87. package/dist/drafts/draft08/messages.js +66 -0
  88. package/dist/drafts/draft08/messages.js.map +1 -0
  89. package/dist/drafts/draft08/rules.d.ts +8 -0
  90. package/dist/drafts/draft08/rules.d.ts.map +1 -0
  91. package/dist/drafts/draft08/rules.js +83 -0
  92. package/dist/drafts/draft08/rules.js.map +1 -0
  93. package/dist/drafts/draft08/session-fsm.d.ts +47 -0
  94. package/dist/drafts/draft08/session-fsm.d.ts.map +1 -0
  95. package/dist/drafts/draft08/session-fsm.js +483 -0
  96. package/dist/drafts/draft08/session-fsm.js.map +1 -0
  97. package/dist/drafts/draft08/session.d.ts +5 -0
  98. package/dist/drafts/draft08/session.d.ts.map +1 -0
  99. package/dist/drafts/draft08/session.js +5 -0
  100. package/dist/drafts/draft08/session.js.map +1 -0
  101. package/dist/drafts/draft08/types.d.ts +268 -0
  102. package/dist/drafts/draft08/types.d.ts.map +1 -0
  103. package/dist/drafts/draft08/types.js +4 -0
  104. package/dist/drafts/draft08/types.js.map +1 -0
  105. package/dist/drafts/draft09/codec.d.ts +21 -0
  106. package/dist/drafts/draft09/codec.d.ts.map +1 -0
  107. package/dist/drafts/draft09/codec.js +721 -0
  108. package/dist/drafts/draft09/codec.js.map +1 -0
  109. package/dist/drafts/draft09/data-streams.d.ts +12 -0
  110. package/dist/drafts/draft09/data-streams.d.ts.map +1 -0
  111. package/dist/drafts/draft09/data-streams.js +307 -0
  112. package/dist/drafts/draft09/data-streams.js.map +1 -0
  113. package/dist/drafts/draft09/index.d.ts +9 -0
  114. package/dist/drafts/draft09/index.d.ts.map +1 -0
  115. package/dist/drafts/draft09/index.js +7 -0
  116. package/dist/drafts/draft09/index.js.map +1 -0
  117. package/dist/drafts/draft09/messages.d.ts +34 -0
  118. package/dist/drafts/draft09/messages.d.ts.map +1 -0
  119. package/dist/drafts/draft09/messages.js +66 -0
  120. package/dist/drafts/draft09/messages.js.map +1 -0
  121. package/dist/drafts/draft09/rules.d.ts +8 -0
  122. package/dist/drafts/draft09/rules.d.ts.map +1 -0
  123. package/dist/drafts/draft09/rules.js +83 -0
  124. package/dist/drafts/draft09/rules.js.map +1 -0
  125. package/dist/drafts/draft09/session-fsm.d.ts +47 -0
  126. package/dist/drafts/draft09/session-fsm.d.ts.map +1 -0
  127. package/dist/drafts/draft09/session-fsm.js +483 -0
  128. package/dist/drafts/draft09/session-fsm.js.map +1 -0
  129. package/dist/drafts/draft09/session.d.ts +5 -0
  130. package/dist/drafts/draft09/session.d.ts.map +1 -0
  131. package/dist/drafts/draft09/session.js +5 -0
  132. package/dist/drafts/draft09/session.js.map +1 -0
  133. package/dist/drafts/draft09/types.d.ts +268 -0
  134. package/dist/drafts/draft09/types.d.ts.map +1 -0
  135. package/dist/drafts/draft09/types.js +4 -0
  136. package/dist/drafts/draft09/types.js.map +1 -0
  137. package/dist/drafts/draft10/codec.d.ts +21 -0
  138. package/dist/drafts/draft10/codec.d.ts.map +1 -0
  139. package/dist/drafts/draft10/codec.js +721 -0
  140. package/dist/drafts/draft10/codec.js.map +1 -0
  141. package/dist/drafts/draft10/data-streams.d.ts +12 -0
  142. package/dist/drafts/draft10/data-streams.d.ts.map +1 -0
  143. package/dist/drafts/draft10/data-streams.js +307 -0
  144. package/dist/drafts/draft10/data-streams.js.map +1 -0
  145. package/dist/drafts/draft10/index.d.ts +9 -0
  146. package/dist/drafts/draft10/index.d.ts.map +1 -0
  147. package/dist/drafts/draft10/index.js +7 -0
  148. package/dist/drafts/draft10/index.js.map +1 -0
  149. package/dist/drafts/draft10/messages.d.ts +34 -0
  150. package/dist/drafts/draft10/messages.d.ts.map +1 -0
  151. package/dist/drafts/draft10/messages.js +66 -0
  152. package/dist/drafts/draft10/messages.js.map +1 -0
  153. package/dist/drafts/draft10/rules.d.ts +8 -0
  154. package/dist/drafts/draft10/rules.d.ts.map +1 -0
  155. package/dist/drafts/draft10/rules.js +83 -0
  156. package/dist/drafts/draft10/rules.js.map +1 -0
  157. package/dist/drafts/draft10/session-fsm.d.ts +47 -0
  158. package/dist/drafts/draft10/session-fsm.d.ts.map +1 -0
  159. package/dist/drafts/draft10/session-fsm.js +483 -0
  160. package/dist/drafts/draft10/session-fsm.js.map +1 -0
  161. package/dist/drafts/draft10/session.d.ts +5 -0
  162. package/dist/drafts/draft10/session.d.ts.map +1 -0
  163. package/dist/drafts/draft10/session.js +5 -0
  164. package/dist/drafts/draft10/session.js.map +1 -0
  165. package/dist/drafts/draft10/types.d.ts +268 -0
  166. package/dist/drafts/draft10/types.d.ts.map +1 -0
  167. package/dist/drafts/draft10/types.js +4 -0
  168. package/dist/drafts/draft10/types.js.map +1 -0
  169. package/dist/drafts/draft11/codec.d.ts +25 -0
  170. package/dist/drafts/draft11/codec.d.ts.map +1 -0
  171. package/dist/drafts/draft11/codec.js +775 -0
  172. package/dist/drafts/draft11/codec.js.map +1 -0
  173. package/dist/drafts/draft11/data-streams.d.ts +10 -0
  174. package/dist/drafts/draft11/data-streams.d.ts.map +1 -0
  175. package/dist/drafts/draft11/data-streams.js +253 -0
  176. package/dist/drafts/draft11/data-streams.js.map +1 -0
  177. package/dist/drafts/draft11/index.d.ts +9 -0
  178. package/dist/drafts/draft11/index.d.ts.map +1 -0
  179. package/dist/drafts/draft11/index.js +7 -0
  180. package/dist/drafts/draft11/index.js.map +1 -0
  181. package/dist/drafts/draft11/messages.d.ts +33 -0
  182. package/dist/drafts/draft11/messages.d.ts.map +1 -0
  183. package/dist/drafts/draft11/messages.js +65 -0
  184. package/dist/drafts/draft11/messages.js.map +1 -0
  185. package/dist/drafts/draft11/rules.d.ts +8 -0
  186. package/dist/drafts/draft11/rules.d.ts.map +1 -0
  187. package/dist/drafts/draft11/rules.js +88 -0
  188. package/dist/drafts/draft11/rules.js.map +1 -0
  189. package/dist/drafts/draft11/session-fsm.d.ts +52 -0
  190. package/dist/drafts/draft11/session-fsm.d.ts.map +1 -0
  191. package/dist/drafts/draft11/session-fsm.js +530 -0
  192. package/dist/drafts/draft11/session-fsm.js.map +1 -0
  193. package/dist/drafts/draft11/session.d.ts +5 -0
  194. package/dist/drafts/draft11/session.d.ts.map +1 -0
  195. package/dist/drafts/draft11/session.js +5 -0
  196. package/dist/drafts/draft11/session.js.map +1 -0
  197. package/dist/drafts/draft11/types.d.ts +266 -0
  198. package/dist/drafts/draft11/types.d.ts.map +1 -0
  199. package/dist/drafts/draft11/types.js +4 -0
  200. package/dist/drafts/draft11/types.js.map +1 -0
  201. package/dist/drafts/draft12/codec.d.ts +27 -0
  202. package/dist/drafts/draft12/codec.d.ts.map +1 -0
  203. package/dist/drafts/draft12/codec.js +918 -0
  204. package/dist/drafts/draft12/codec.js.map +1 -0
  205. package/dist/drafts/draft12/data-streams.d.ts +10 -0
  206. package/dist/drafts/draft12/data-streams.d.ts.map +1 -0
  207. package/dist/drafts/draft12/data-streams.js +254 -0
  208. package/dist/drafts/draft12/data-streams.js.map +1 -0
  209. package/dist/drafts/draft12/index.d.ts +9 -0
  210. package/dist/drafts/draft12/index.d.ts.map +1 -0
  211. package/dist/drafts/draft12/index.js +7 -0
  212. package/dist/drafts/draft12/index.js.map +1 -0
  213. package/dist/drafts/draft12/messages.d.ts +37 -0
  214. package/dist/drafts/draft12/messages.d.ts.map +1 -0
  215. package/dist/drafts/draft12/messages.js +77 -0
  216. package/dist/drafts/draft12/messages.js.map +1 -0
  217. package/dist/drafts/draft12/rules.d.ts +8 -0
  218. package/dist/drafts/draft12/rules.d.ts.map +1 -0
  219. package/dist/drafts/draft12/rules.js +94 -0
  220. package/dist/drafts/draft12/rules.js.map +1 -0
  221. package/dist/drafts/draft12/session-fsm.d.ts +55 -0
  222. package/dist/drafts/draft12/session-fsm.d.ts.map +1 -0
  223. package/dist/drafts/draft12/session-fsm.js +569 -0
  224. package/dist/drafts/draft12/session-fsm.js.map +1 -0
  225. package/dist/drafts/draft12/session.d.ts +5 -0
  226. package/dist/drafts/draft12/session.d.ts.map +1 -0
  227. package/dist/drafts/draft12/session.js +5 -0
  228. package/dist/drafts/draft12/session.js.map +1 -0
  229. package/dist/drafts/draft12/types.d.ts +294 -0
  230. package/dist/drafts/draft12/types.d.ts.map +1 -0
  231. package/dist/drafts/draft12/types.js +4 -0
  232. package/dist/drafts/draft12/types.js.map +1 -0
  233. package/dist/drafts/draft13/codec.d.ts +27 -0
  234. package/dist/drafts/draft13/codec.d.ts.map +1 -0
  235. package/dist/drafts/draft13/codec.js +1000 -0
  236. package/dist/drafts/draft13/codec.js.map +1 -0
  237. package/dist/drafts/draft13/data-streams.d.ts +10 -0
  238. package/dist/drafts/draft13/data-streams.d.ts.map +1 -0
  239. package/dist/drafts/draft13/data-streams.js +254 -0
  240. package/dist/drafts/draft13/data-streams.js.map +1 -0
  241. package/dist/drafts/draft13/index.d.ts +9 -0
  242. package/dist/drafts/draft13/index.d.ts.map +1 -0
  243. package/dist/drafts/draft13/index.js +7 -0
  244. package/dist/drafts/draft13/index.js.map +1 -0
  245. package/dist/drafts/draft13/messages.d.ts +38 -0
  246. package/dist/drafts/draft13/messages.d.ts.map +1 -0
  247. package/dist/drafts/draft13/messages.js +79 -0
  248. package/dist/drafts/draft13/messages.js.map +1 -0
  249. package/dist/drafts/draft13/rules.d.ts +8 -0
  250. package/dist/drafts/draft13/rules.d.ts.map +1 -0
  251. package/dist/drafts/draft13/rules.js +96 -0
  252. package/dist/drafts/draft13/rules.js.map +1 -0
  253. package/dist/drafts/draft13/session-fsm.d.ts +56 -0
  254. package/dist/drafts/draft13/session-fsm.d.ts.map +1 -0
  255. package/dist/drafts/draft13/session-fsm.js +581 -0
  256. package/dist/drafts/draft13/session-fsm.js.map +1 -0
  257. package/dist/drafts/draft13/session.d.ts +5 -0
  258. package/dist/drafts/draft13/session.d.ts.map +1 -0
  259. package/dist/drafts/draft13/session.js +5 -0
  260. package/dist/drafts/draft13/session.js.map +1 -0
  261. package/dist/drafts/draft13/types.d.ts +310 -0
  262. package/dist/drafts/draft13/types.d.ts.map +1 -0
  263. package/dist/drafts/draft13/types.js +4 -0
  264. package/dist/drafts/draft13/types.js.map +1 -0
  265. package/dist/drafts/draft14/codec.d.ts +34 -0
  266. package/dist/drafts/draft14/codec.d.ts.map +1 -0
  267. package/dist/drafts/draft14/codec.js +752 -0
  268. package/dist/drafts/draft14/codec.js.map +1 -0
  269. package/dist/drafts/draft14/data-streams.d.ts +56 -0
  270. package/dist/drafts/draft14/data-streams.d.ts.map +1 -0
  271. package/dist/drafts/draft14/data-streams.js +729 -0
  272. package/dist/drafts/draft14/data-streams.js.map +1 -0
  273. package/dist/drafts/draft14/index.d.ts +9 -0
  274. package/dist/drafts/draft14/index.d.ts.map +1 -0
  275. package/dist/drafts/draft14/index.js +7 -0
  276. package/dist/drafts/draft14/index.js.map +1 -0
  277. package/dist/drafts/draft14/messages.d.ts +36 -0
  278. package/dist/drafts/draft14/messages.d.ts.map +1 -0
  279. package/dist/drafts/draft14/messages.js +71 -0
  280. package/dist/drafts/draft14/messages.js.map +1 -0
  281. package/dist/drafts/draft14/rules.d.ts +8 -0
  282. package/dist/drafts/draft14/rules.d.ts.map +1 -0
  283. package/dist/drafts/draft14/rules.js +101 -0
  284. package/dist/drafts/draft14/rules.js.map +1 -0
  285. package/dist/drafts/draft14/session-fsm.d.ts +58 -0
  286. package/dist/drafts/draft14/session-fsm.d.ts.map +1 -0
  287. package/dist/drafts/draft14/session-fsm.js +648 -0
  288. package/dist/drafts/draft14/session-fsm.js.map +1 -0
  289. package/dist/drafts/draft14/session.d.ts +5 -0
  290. package/dist/drafts/draft14/session.d.ts.map +1 -0
  291. package/dist/drafts/draft14/session.js +5 -0
  292. package/dist/drafts/draft14/session.js.map +1 -0
  293. package/dist/drafts/draft14/types.d.ts +263 -0
  294. package/dist/drafts/draft14/types.d.ts.map +1 -0
  295. package/dist/drafts/draft14/types.js +4 -0
  296. package/dist/drafts/draft14/types.js.map +1 -0
  297. package/dist/drafts/draft15/codec.d.ts +33 -0
  298. package/dist/drafts/draft15/codec.d.ts.map +1 -0
  299. package/dist/drafts/draft15/codec.js +742 -0
  300. package/dist/drafts/draft15/codec.js.map +1 -0
  301. package/dist/drafts/draft15/data-streams.d.ts +45 -0
  302. package/dist/drafts/draft15/data-streams.d.ts.map +1 -0
  303. package/dist/drafts/draft15/data-streams.js +675 -0
  304. package/dist/drafts/draft15/data-streams.js.map +1 -0
  305. package/dist/drafts/draft15/index.d.ts +9 -0
  306. package/dist/drafts/draft15/index.d.ts.map +1 -0
  307. package/dist/drafts/draft15/index.js +7 -0
  308. package/dist/drafts/draft15/index.js.map +1 -0
  309. package/dist/drafts/draft15/messages.d.ts +31 -0
  310. package/dist/drafts/draft15/messages.d.ts.map +1 -0
  311. package/dist/drafts/draft15/messages.js +59 -0
  312. package/dist/drafts/draft15/messages.js.map +1 -0
  313. package/dist/drafts/draft15/rules.d.ts +8 -0
  314. package/dist/drafts/draft15/rules.d.ts.map +1 -0
  315. package/dist/drafts/draft15/rules.js +83 -0
  316. package/dist/drafts/draft15/rules.js.map +1 -0
  317. package/dist/drafts/draft15/session-fsm.d.ts +48 -0
  318. package/dist/drafts/draft15/session-fsm.d.ts.map +1 -0
  319. package/dist/drafts/draft15/session-fsm.js +479 -0
  320. package/dist/drafts/draft15/session-fsm.js.map +1 -0
  321. package/dist/drafts/draft15/session.d.ts +5 -0
  322. package/dist/drafts/draft15/session.d.ts.map +1 -0
  323. package/dist/drafts/draft15/session.js +5 -0
  324. package/dist/drafts/draft15/session.js.map +1 -0
  325. package/dist/drafts/draft15/types.d.ts +232 -0
  326. package/dist/drafts/draft15/types.d.ts.map +1 -0
  327. package/dist/drafts/draft15/types.js +4 -0
  328. package/dist/drafts/draft15/types.js.map +1 -0
  329. package/dist/drafts/draft16/codec.d.ts +29 -0
  330. package/dist/drafts/draft16/codec.d.ts.map +1 -0
  331. package/dist/drafts/draft16/codec.js +747 -0
  332. package/dist/drafts/draft16/codec.js.map +1 -0
  333. package/dist/drafts/draft16/data-streams.d.ts +34 -0
  334. package/dist/drafts/draft16/data-streams.d.ts.map +1 -0
  335. package/dist/drafts/draft16/data-streams.js +667 -0
  336. package/dist/drafts/draft16/data-streams.js.map +1 -0
  337. package/dist/drafts/draft16/index.d.ts +9 -0
  338. package/dist/drafts/draft16/index.d.ts.map +1 -0
  339. package/dist/drafts/draft16/index.js +7 -0
  340. package/dist/drafts/draft16/index.js.map +1 -0
  341. package/dist/drafts/draft16/messages.d.ts +32 -0
  342. package/dist/drafts/draft16/messages.d.ts.map +1 -0
  343. package/dist/drafts/draft16/messages.js +62 -0
  344. package/dist/drafts/draft16/messages.js.map +1 -0
  345. package/dist/drafts/draft16/rules.d.ts +8 -0
  346. package/dist/drafts/draft16/rules.d.ts.map +1 -0
  347. package/dist/drafts/draft16/rules.js +84 -0
  348. package/dist/drafts/draft16/rules.js.map +1 -0
  349. package/dist/drafts/draft16/session-fsm.d.ts +48 -0
  350. package/dist/drafts/draft16/session-fsm.d.ts.map +1 -0
  351. package/dist/drafts/draft16/session-fsm.js +474 -0
  352. package/dist/drafts/draft16/session-fsm.js.map +1 -0
  353. package/dist/drafts/draft16/session.d.ts +5 -0
  354. package/dist/drafts/draft16/session.d.ts.map +1 -0
  355. package/dist/drafts/draft16/session.js +5 -0
  356. package/dist/drafts/draft16/session.js.map +1 -0
  357. package/dist/drafts/draft16/types.d.ts +238 -0
  358. package/dist/drafts/draft16/types.d.ts.map +1 -0
  359. package/dist/drafts/draft16/types.js +4 -0
  360. package/dist/drafts/draft16/types.js.map +1 -0
  361. package/dist/drafts/draft17/codec.d.ts +29 -0
  362. package/dist/drafts/draft17/codec.d.ts.map +1 -0
  363. package/dist/drafts/draft17/codec.js +799 -0
  364. package/dist/drafts/draft17/codec.js.map +1 -0
  365. package/dist/drafts/draft17/data-streams.d.ts +13 -0
  366. package/dist/drafts/draft17/data-streams.d.ts.map +1 -0
  367. package/dist/drafts/draft17/data-streams.js +633 -0
  368. package/dist/drafts/draft17/data-streams.js.map +1 -0
  369. package/dist/drafts/draft17/index.d.ts +8 -0
  370. package/dist/drafts/draft17/index.d.ts.map +1 -0
  371. package/dist/drafts/draft17/index.js +6 -0
  372. package/dist/drafts/draft17/index.js.map +1 -0
  373. package/dist/drafts/draft17/messages.d.ts +25 -0
  374. package/dist/drafts/draft17/messages.d.ts.map +1 -0
  375. package/dist/drafts/draft17/messages.js +48 -0
  376. package/dist/drafts/draft17/messages.js.map +1 -0
  377. package/dist/drafts/draft17/rules.d.ts +8 -0
  378. package/dist/drafts/draft17/rules.d.ts.map +1 -0
  379. package/dist/drafts/draft17/rules.js +71 -0
  380. package/dist/drafts/draft17/rules.js.map +1 -0
  381. package/dist/drafts/draft17/session-fsm.d.ts +45 -0
  382. package/dist/drafts/draft17/session-fsm.d.ts.map +1 -0
  383. package/dist/drafts/draft17/session-fsm.js +328 -0
  384. package/dist/drafts/draft17/session-fsm.js.map +1 -0
  385. package/dist/drafts/draft17/session.d.ts +5 -0
  386. package/dist/drafts/draft17/session.d.ts.map +1 -0
  387. package/dist/drafts/draft17/session.js +6 -0
  388. package/dist/drafts/draft17/session.js.map +1 -0
  389. package/dist/drafts/draft17/types.d.ts +219 -0
  390. package/dist/drafts/draft17/types.d.ts.map +1 -0
  391. package/dist/drafts/draft17/types.js +3 -0
  392. package/dist/drafts/draft17/types.js.map +1 -0
  393. package/dist/index.d.ts +46 -38
  394. package/dist/index.d.ts.map +1 -0
  395. package/dist/index.js +93 -110
  396. package/dist/index.js.map +1 -0
  397. package/dist/session.d.ts +4 -8
  398. package/dist/session.d.ts.map +1 -0
  399. package/dist/session.js +32 -26
  400. package/dist/session.js.map +1 -0
  401. package/package.json +2 -2
  402. package/src/core/buffer-reader.ts +1 -1
  403. package/src/core/hex.ts +17 -0
  404. package/src/drafts/draft07/codec.ts +933 -991
  405. package/src/drafts/draft07/data-streams.ts +240 -0
  406. package/src/drafts/draft07/index.ts +89 -69
  407. package/src/drafts/draft07/messages.ts +42 -44
  408. package/src/drafts/draft07/rules.ts +101 -104
  409. package/src/drafts/draft07/types.ts +72 -0
  410. package/src/drafts/draft08/codec.ts +944 -1254
  411. package/src/drafts/draft08/data-streams.ts +359 -0
  412. package/src/drafts/draft08/types.ts +384 -377
  413. package/src/drafts/draft09/codec.ts +936 -1235
  414. package/src/drafts/draft09/data-streams.ts +332 -0
  415. package/src/drafts/draft09/types.ts +384 -376
  416. package/src/drafts/draft10/codec.ts +936 -1235
  417. package/src/drafts/draft10/data-streams.ts +332 -0
  418. package/src/drafts/draft10/types.ts +384 -376
  419. package/src/drafts/draft11/codec.ts +979 -1198
  420. package/src/drafts/draft11/data-streams.ts +269 -0
  421. package/src/drafts/draft11/types.ts +381 -375
  422. package/src/drafts/draft12/codec.ts +1113 -1354
  423. package/src/drafts/draft12/data-streams.ts +275 -0
  424. package/src/drafts/draft12/types.ts +419 -414
  425. package/src/drafts/draft13/codec.ts +1197 -1438
  426. package/src/drafts/draft13/data-streams.ts +275 -0
  427. package/src/drafts/draft13/types.ts +438 -433
  428. package/src/drafts/draft14/codec.ts +1034 -1480
  429. package/src/drafts/draft14/data-streams.ts +798 -0
  430. package/src/drafts/draft14/types.ts +381 -365
  431. package/src/drafts/draft15/codec.ts +969 -1661
  432. package/src/drafts/draft15/data-streams.ts +778 -0
  433. package/src/drafts/draft15/types.ts +339 -336
  434. package/src/drafts/draft16/codec.ts +957 -1623
  435. package/src/drafts/draft16/data-streams.ts +773 -0
  436. package/src/drafts/draft16/types.ts +357 -354
  437. package/src/drafts/draft17/codec.ts +962 -1621
  438. package/src/drafts/draft17/data-streams.ts +742 -0
  439. package/src/drafts/draft17/types.ts +313 -310
@@ -1,1480 +1,1034 @@
1
- import { BufferReader } from "../../core/buffer-reader.js";
2
- import { BufferWriter } from "../../core/buffer-writer.js";
3
- import type { BaseCodec, DecodeResult } from "../../core/types.js";
4
- import { DecodeError } from "../../core/types.js";
5
- import {
6
- MESSAGE_ID_MAP,
7
- MSG_CLIENT_SETUP,
8
- MSG_FETCH,
9
- MSG_FETCH_CANCEL,
10
- MSG_FETCH_ERROR,
11
- MSG_FETCH_OK,
12
- MSG_GOAWAY,
13
- MSG_MAX_REQUEST_ID,
14
- MSG_PUBLISH,
15
- MSG_PUBLISH_DONE,
16
- MSG_PUBLISH_ERROR,
17
- MSG_PUBLISH_NAMESPACE,
18
- MSG_PUBLISH_NAMESPACE_CANCEL,
19
- MSG_PUBLISH_NAMESPACE_DONE,
20
- MSG_PUBLISH_NAMESPACE_ERROR,
21
- MSG_PUBLISH_NAMESPACE_OK,
22
- MSG_PUBLISH_OK,
23
- MSG_REQUESTS_BLOCKED,
24
- MSG_SERVER_SETUP,
25
- MSG_SUBSCRIBE,
26
- MSG_SUBSCRIBE_ERROR,
27
- MSG_SUBSCRIBE_NAMESPACE,
28
- MSG_SUBSCRIBE_NAMESPACE_ERROR,
29
- MSG_SUBSCRIBE_NAMESPACE_OK,
30
- MSG_SUBSCRIBE_OK,
31
- MSG_SUBSCRIBE_UPDATE,
32
- MSG_TRACK_STATUS,
33
- MSG_TRACK_STATUS_ERROR,
34
- MSG_TRACK_STATUS_OK,
35
- MSG_UNSUBSCRIBE,
36
- MSG_UNSUBSCRIBE_NAMESPACE,
37
- PARAM_MAX_REQUEST_ID,
38
- PARAM_PATH,
39
- PARAM_ROLE,
40
- } from "./messages.js";
41
- import type {
42
- DatagramObject,
43
- DataStreamEvent,
44
- Draft14DataStream,
45
- Draft14Message,
46
- Draft14Params,
47
- FetchStream,
48
- FetchStreamHeader,
49
- ObjectPayload,
50
- SubgroupStream,
51
- SubgroupStreamHeader,
52
- UnknownParam,
53
- } from "./types.js";
54
-
55
- // ─── Helpers ───────────────────────────────────────────────────────────────────
56
-
57
- function bytesToHex(bytes: Uint8Array): string {
58
- let hex = "";
59
- for (let i = 0; i < bytes.byteLength; i++) {
60
- hex += (bytes[i] as number).toString(16).padStart(2, "0");
61
- }
62
- return hex;
63
- }
64
-
65
- function hexToBytes(hex: string): Uint8Array {
66
- const bytes = new Uint8Array(hex.length / 2);
67
- for (let i = 0; i < hex.length; i += 2) {
68
- bytes[i / 2] = parseInt(hex.substring(i, i + 2), 16);
69
- }
70
- return bytes;
71
- }
72
-
73
- // ─── Parameter Encoding/Decoding ───────────────────────────────────────────────
74
-
75
- function encodeParams(params: Draft14Params, writer: BufferWriter): void {
76
- // Count total params
77
- let count = 0;
78
- if (params.role !== undefined) count++;
79
- if (params.path !== undefined) count++;
80
- if (params.max_request_id !== undefined) count++;
81
- if (params.unknown) count += params.unknown.length;
82
-
83
- writer.writeVarInt(count);
84
-
85
- // ROLE (0x00) - even, varint value
86
- if (params.role !== undefined) {
87
- writer.writeVarInt(PARAM_ROLE);
88
- writer.writeVarInt(params.role);
89
- }
90
-
91
- // PATH (0x01) - odd, length-prefixed bytes
92
- if (params.path !== undefined) {
93
- writer.writeVarInt(PARAM_PATH);
94
- const encoded = new TextEncoder().encode(params.path);
95
- writer.writeVarInt(encoded.byteLength);
96
- writer.writeBytes(encoded);
97
- }
98
-
99
- // MAX_REQUEST_ID (0x02) - even, varint value
100
- if (params.max_request_id !== undefined) {
101
- writer.writeVarInt(PARAM_MAX_REQUEST_ID);
102
- writer.writeVarInt(params.max_request_id);
103
- }
104
-
105
- // Unknown params
106
- if (params.unknown) {
107
- for (const u of params.unknown) {
108
- const id = BigInt(u.id);
109
- writer.writeVarInt(id);
110
- const raw = hexToBytes(u.raw_hex);
111
- writer.writeVarInt(raw.byteLength);
112
- writer.writeBytes(raw);
113
- }
114
- }
115
- }
116
-
117
- function decodeParams(reader: BufferReader): Draft14Params {
118
- const count = Number(reader.readVarInt());
119
- const result: Draft14Params = {};
120
- const unknown: UnknownParam[] = [];
121
-
122
- for (let i = 0; i < count; i++) {
123
- const paramType = reader.readVarInt();
124
-
125
- if (paramType % 2n === 0n) {
126
- // Even: value is a varint directly
127
- const value = reader.readVarInt();
128
- if (paramType === PARAM_ROLE) {
129
- result.role = value;
130
- } else if (paramType === PARAM_MAX_REQUEST_ID) {
131
- result.max_request_id = value;
132
- } else {
133
- // Unknown even param — encode the varint value as raw bytes
134
- const tmpWriter = new BufferWriter(16);
135
- tmpWriter.writeVarInt(value);
136
- const raw = tmpWriter.finish();
137
- unknown.push({
138
- id: `0x${paramType.toString(16)}`,
139
- length: raw.byteLength,
140
- raw_hex: bytesToHex(raw),
141
- });
142
- }
143
- } else {
144
- // Odd: value is length-prefixed bytes
145
- const length = Number(reader.readVarInt());
146
- const bytes = reader.readBytes(length);
147
- if (paramType === PARAM_PATH) {
148
- result.path = new TextDecoder().decode(bytes);
149
- } else {
150
- unknown.push({
151
- id: `0x${paramType.toString(16)}`,
152
- length,
153
- raw_hex: bytesToHex(bytes),
154
- });
155
- }
156
- }
157
- }
158
-
159
- if (unknown.length > 0) {
160
- result.unknown = unknown;
161
- }
162
-
163
- return result;
164
- }
165
-
166
- // ─── Payload Encoders ──────────────────────────────────────────────────────────
167
-
168
- function encodeClientSetupPayload(
169
- msg: Draft14Message & { type: "client_setup" },
170
- w: BufferWriter,
171
- ): void {
172
- w.writeVarInt(msg.supported_versions.length);
173
- for (const v of msg.supported_versions) {
174
- w.writeVarInt(v);
175
- }
176
- encodeParams(msg.parameters, w);
177
- }
178
-
179
- function encodeServerSetupPayload(
180
- msg: Draft14Message & { type: "server_setup" },
181
- w: BufferWriter,
182
- ): void {
183
- w.writeVarInt(msg.selected_version);
184
- encodeParams(msg.parameters, w);
185
- }
186
-
187
- function encodeSubscribePayload(
188
- msg: Draft14Message & { type: "subscribe" },
189
- w: BufferWriter,
190
- ): void {
191
- w.writeVarInt(msg.request_id);
192
- w.writeTuple(msg.track_namespace);
193
- w.writeString(msg.track_name);
194
- w.writeUint8(Number(msg.subscriber_priority));
195
- w.writeVarInt(msg.group_order);
196
- w.writeVarInt(msg.forward);
197
- w.writeVarInt(msg.filter_type);
198
- const ft = Number(msg.filter_type);
199
- if (ft >= 3) {
200
- w.writeVarInt(msg.start_group!);
201
- w.writeVarInt(msg.start_object!);
202
- }
203
- if (ft === 4) {
204
- w.writeVarInt(msg.end_group!);
205
- }
206
- encodeParams(msg.parameters, w);
207
- }
208
-
209
- function encodeSubscribeOkPayload(
210
- msg: Draft14Message & { type: "subscribe_ok" },
211
- w: BufferWriter,
212
- ): void {
213
- w.writeVarInt(msg.request_id);
214
- w.writeVarInt(msg.track_alias);
215
- w.writeVarInt(msg.expires);
216
- w.writeVarInt(msg.group_order);
217
- w.writeVarInt(msg.content_exists);
218
- if (Number(msg.content_exists) === 1) {
219
- w.writeVarInt(msg.largest_group!);
220
- w.writeVarInt(msg.largest_object!);
221
- }
222
- encodeParams(msg.parameters, w);
223
- }
224
-
225
- function encodeSubscribeUpdatePayload(
226
- msg: Draft14Message & { type: "subscribe_update" },
227
- w: BufferWriter,
228
- ): void {
229
- w.writeVarInt(msg.request_id);
230
- w.writeVarInt(msg.start_group);
231
- w.writeVarInt(msg.start_object);
232
- w.writeVarInt(msg.end_group);
233
- w.writeUint8(Number(msg.subscriber_priority));
234
- w.writeVarInt(msg.forward);
235
- encodeParams(msg.parameters, w);
236
- }
237
-
238
- function encodeSubscribeErrorPayload(
239
- msg: Draft14Message & { type: "subscribe_error" },
240
- w: BufferWriter,
241
- ): void {
242
- w.writeVarInt(msg.request_id);
243
- w.writeVarInt(msg.error_code);
244
- w.writeString(msg.reason_phrase);
245
- }
246
-
247
- function encodeUnsubscribePayload(
248
- msg: Draft14Message & { type: "unsubscribe" },
249
- w: BufferWriter,
250
- ): void {
251
- w.writeVarInt(msg.request_id);
252
- }
253
-
254
- function encodePublishPayload(msg: Draft14Message & { type: "publish" }, w: BufferWriter): void {
255
- w.writeVarInt(msg.request_id);
256
- w.writeTuple(msg.track_namespace);
257
- w.writeString(msg.track_name);
258
- w.writeVarInt(msg.forward);
259
- encodeParams(msg.parameters, w);
260
- }
261
-
262
- function encodePublishOkPayload(
263
- msg: Draft14Message & { type: "publish_ok" },
264
- w: BufferWriter,
265
- ): void {
266
- w.writeVarInt(msg.request_id);
267
- w.writeVarInt(msg.track_alias);
268
- w.writeVarInt(msg.forward);
269
- encodeParams(msg.parameters, w);
270
- }
271
-
272
- function encodePublishErrorPayload(
273
- msg: Draft14Message & { type: "publish_error" },
274
- w: BufferWriter,
275
- ): void {
276
- w.writeVarInt(msg.request_id);
277
- w.writeVarInt(msg.error_code);
278
- w.writeString(msg.reason_phrase);
279
- }
280
-
281
- function encodePublishDonePayload(
282
- msg: Draft14Message & { type: "publish_done" },
283
- w: BufferWriter,
284
- ): void {
285
- w.writeVarInt(msg.request_id);
286
- w.writeVarInt(msg.status_code);
287
- w.writeString(msg.reason_phrase);
288
- }
289
-
290
- function encodePublishNamespacePayload(
291
- msg: Draft14Message & { type: "publish_namespace" },
292
- w: BufferWriter,
293
- ): void {
294
- w.writeVarInt(msg.request_id);
295
- w.writeTuple(msg.track_namespace);
296
- encodeParams(msg.parameters, w);
297
- }
298
-
299
- function encodePublishNamespaceOkPayload(
300
- msg: Draft14Message & { type: "publish_namespace_ok" },
301
- w: BufferWriter,
302
- ): void {
303
- w.writeVarInt(msg.request_id);
304
- encodeParams(msg.parameters, w);
305
- }
306
-
307
- function encodePublishNamespaceErrorPayload(
308
- msg: Draft14Message & { type: "publish_namespace_error" },
309
- w: BufferWriter,
310
- ): void {
311
- w.writeVarInt(msg.request_id);
312
- w.writeVarInt(msg.error_code);
313
- w.writeString(msg.reason_phrase);
314
- }
315
-
316
- function encodePublishNamespaceDonePayload(
317
- msg: Draft14Message & { type: "publish_namespace_done" },
318
- w: BufferWriter,
319
- ): void {
320
- w.writeVarInt(msg.request_id);
321
- w.writeVarInt(msg.status_code);
322
- w.writeString(msg.reason_phrase);
323
- }
324
-
325
- function encodePublishNamespaceCancelPayload(
326
- msg: Draft14Message & { type: "publish_namespace_cancel" },
327
- w: BufferWriter,
328
- ): void {
329
- w.writeVarInt(msg.request_id);
330
- }
331
-
332
- function encodeSubscribeNamespacePayload(
333
- msg: Draft14Message & { type: "subscribe_namespace" },
334
- w: BufferWriter,
335
- ): void {
336
- w.writeVarInt(msg.request_id);
337
- w.writeTuple(msg.namespace_prefix);
338
- encodeParams(msg.parameters, w);
339
- }
340
-
341
- function encodeSubscribeNamespaceOkPayload(
342
- msg: Draft14Message & { type: "subscribe_namespace_ok" },
343
- w: BufferWriter,
344
- ): void {
345
- w.writeVarInt(msg.request_id);
346
- encodeParams(msg.parameters, w);
347
- }
348
-
349
- function encodeSubscribeNamespaceErrorPayload(
350
- msg: Draft14Message & { type: "subscribe_namespace_error" },
351
- w: BufferWriter,
352
- ): void {
353
- w.writeVarInt(msg.request_id);
354
- w.writeVarInt(msg.error_code);
355
- w.writeString(msg.reason_phrase);
356
- }
357
-
358
- function encodeUnsubscribeNamespacePayload(
359
- msg: Draft14Message & { type: "unsubscribe_namespace" },
360
- w: BufferWriter,
361
- ): void {
362
- w.writeVarInt(msg.request_id);
363
- }
364
-
365
- function encodeFetchPayload(msg: Draft14Message & { type: "fetch" }, w: BufferWriter): void {
366
- w.writeVarInt(msg.request_id);
367
- w.writeTuple(msg.track_namespace);
368
- w.writeString(msg.track_name);
369
- w.writeVarInt(msg.start_group);
370
- w.writeVarInt(msg.start_object);
371
- w.writeVarInt(msg.end_group);
372
- encodeParams(msg.parameters, w);
373
- }
374
-
375
- function encodeFetchOkPayload(msg: Draft14Message & { type: "fetch_ok" }, w: BufferWriter): void {
376
- w.writeVarInt(msg.request_id);
377
- w.writeVarInt(msg.track_alias);
378
- w.writeVarInt(msg.end_of_track);
379
- encodeParams(msg.parameters, w);
380
- }
381
-
382
- function encodeFetchErrorPayload(
383
- msg: Draft14Message & { type: "fetch_error" },
384
- w: BufferWriter,
385
- ): void {
386
- w.writeVarInt(msg.request_id);
387
- w.writeVarInt(msg.error_code);
388
- w.writeString(msg.reason_phrase);
389
- }
390
-
391
- function encodeFetchCancelPayload(
392
- msg: Draft14Message & { type: "fetch_cancel" },
393
- w: BufferWriter,
394
- ): void {
395
- w.writeVarInt(msg.request_id);
396
- }
397
-
398
- function encodeTrackStatusPayload(
399
- msg: Draft14Message & { type: "track_status" },
400
- w: BufferWriter,
401
- ): void {
402
- w.writeVarInt(msg.request_id);
403
- w.writeTuple(msg.track_namespace);
404
- w.writeString(msg.track_name);
405
- encodeParams(msg.parameters, w);
406
- }
407
-
408
- function encodeTrackStatusOkPayload(
409
- msg: Draft14Message & { type: "track_status_ok" },
410
- w: BufferWriter,
411
- ): void {
412
- w.writeVarInt(msg.request_id);
413
- w.writeVarInt(msg.status_code);
414
- const sc = Number(msg.status_code);
415
- if (sc === 0 || sc === 3) {
416
- w.writeVarInt(msg.largest_group!);
417
- w.writeVarInt(msg.largest_object!);
418
- }
419
- encodeParams(msg.parameters, w);
420
- }
421
-
422
- function encodeTrackStatusErrorPayload(
423
- msg: Draft14Message & { type: "track_status_error" },
424
- w: BufferWriter,
425
- ): void {
426
- w.writeVarInt(msg.request_id);
427
- w.writeVarInt(msg.error_code);
428
- w.writeString(msg.reason_phrase);
429
- }
430
-
431
- function encodeGoAwayPayload(msg: Draft14Message & { type: "goaway" }, w: BufferWriter): void {
432
- w.writeString(msg.new_session_uri);
433
- }
434
-
435
- function encodeMaxRequestIdPayload(
436
- msg: Draft14Message & { type: "max_request_id" },
437
- w: BufferWriter,
438
- ): void {
439
- w.writeVarInt(msg.request_id);
440
- }
441
-
442
- function encodeRequestsBlockedPayload(
443
- msg: Draft14Message & { type: "requests_blocked" },
444
- w: BufferWriter,
445
- ): void {
446
- w.writeVarInt(msg.request_id);
447
- }
448
-
449
- // ─── Payload Decoders ──────────────────────────────────────────────────────────
450
-
451
- function decodeClientSetupPayload(r: BufferReader): Draft14Message {
452
- const numVersions = Number(r.readVarInt());
453
- if (numVersions === 0) {
454
- throw new DecodeError(
455
- "CONSTRAINT_VIOLATION",
456
- "CLIENT_SETUP must offer at least one version",
457
- r.offset,
458
- );
459
- }
460
- const supported_versions: bigint[] = [];
461
- for (let i = 0; i < numVersions; i++) {
462
- supported_versions.push(r.readVarInt());
463
- }
464
- const parameters = decodeParams(r);
465
- return { type: "client_setup", supported_versions, parameters };
466
- }
467
-
468
- function decodeServerSetupPayload(r: BufferReader): Draft14Message {
469
- const selected_version = r.readVarInt();
470
- const parameters = decodeParams(r);
471
- return { type: "server_setup", selected_version, parameters };
472
- }
473
-
474
- function decodeSubscribePayload(r: BufferReader): Draft14Message {
475
- const request_id = r.readVarInt();
476
- const track_namespace = r.readTuple();
477
- const track_name = r.readString();
478
- const subscriber_priority = BigInt(r.readUint8());
479
- const group_order = r.readVarInt();
480
- const forward = r.readVarInt();
481
- const filter_type = r.readVarInt();
482
- const ft = Number(filter_type);
483
-
484
- if (ft < 1 || ft > 4) {
485
- throw new DecodeError("CONSTRAINT_VIOLATION", `Invalid filter_type: ${ft}`, r.offset);
486
- }
487
-
488
- let start_group: bigint | undefined;
489
- let start_object: bigint | undefined;
490
- let end_group: bigint | undefined;
491
-
492
- if (ft >= 3) {
493
- start_group = r.readVarInt();
494
- start_object = r.readVarInt();
495
- }
496
- if (ft === 4) {
497
- end_group = r.readVarInt();
498
- }
499
-
500
- const parameters = decodeParams(r);
501
-
502
- const msg: Draft14Message & { type: "subscribe" } = {
503
- type: "subscribe",
504
- request_id,
505
- track_namespace,
506
- track_name,
507
- subscriber_priority,
508
- group_order,
509
- forward,
510
- filter_type,
511
- parameters,
512
- };
513
-
514
- if (start_group !== undefined) {
515
- return { ...msg, start_group, start_object, end_group } as Draft14Message;
516
- }
517
-
518
- return msg;
519
- }
520
-
521
- function decodeSubscribeOkPayload(r: BufferReader): Draft14Message {
522
- const request_id = r.readVarInt();
523
- const track_alias = r.readVarInt();
524
- const expires = r.readVarInt();
525
- const group_order = r.readVarInt();
526
- const content_exists = r.readVarInt();
527
-
528
- let largest_group: bigint | undefined;
529
- let largest_object: bigint | undefined;
530
-
531
- if (Number(content_exists) === 1) {
532
- largest_group = r.readVarInt();
533
- largest_object = r.readVarInt();
534
- }
535
-
536
- const parameters = decodeParams(r);
537
-
538
- const msg: Draft14Message = {
539
- type: "subscribe_ok",
540
- request_id,
541
- track_alias,
542
- expires,
543
- group_order,
544
- content_exists,
545
- parameters,
546
- ...(largest_group !== undefined ? { largest_group, largest_object } : {}),
547
- } as Draft14Message;
548
-
549
- return msg;
550
- }
551
-
552
- function decodeSubscribeUpdatePayload(r: BufferReader): Draft14Message {
553
- const request_id = r.readVarInt();
554
- const start_group = r.readVarInt();
555
- const start_object = r.readVarInt();
556
- const end_group = r.readVarInt();
557
- const subscriber_priority = BigInt(r.readUint8());
558
- const forward = r.readVarInt();
559
- const parameters = decodeParams(r);
560
- return {
561
- type: "subscribe_update",
562
- request_id,
563
- start_group,
564
- start_object,
565
- end_group,
566
- subscriber_priority,
567
- forward,
568
- parameters,
569
- };
570
- }
571
-
572
- function decodeSubscribeErrorPayload(r: BufferReader): Draft14Message {
573
- const request_id = r.readVarInt();
574
- const error_code = r.readVarInt();
575
- const reason_phrase = r.readString();
576
- return { type: "subscribe_error", request_id, error_code, reason_phrase };
577
- }
578
-
579
- function decodeUnsubscribePayload(r: BufferReader): Draft14Message {
580
- const request_id = r.readVarInt();
581
- return { type: "unsubscribe", request_id };
582
- }
583
-
584
- function decodePublishPayload(r: BufferReader): Draft14Message {
585
- const request_id = r.readVarInt();
586
- const track_namespace = r.readTuple();
587
- const track_name = r.readString();
588
- const forward = r.readVarInt();
589
- const parameters = decodeParams(r);
590
- return { type: "publish", request_id, track_namespace, track_name, forward, parameters };
591
- }
592
-
593
- function decodePublishOkPayload(r: BufferReader): Draft14Message {
594
- const request_id = r.readVarInt();
595
- const track_alias = r.readVarInt();
596
- const forward = r.readVarInt();
597
- const parameters = decodeParams(r);
598
- return { type: "publish_ok", request_id, track_alias, forward, parameters };
599
- }
600
-
601
- function decodePublishErrorPayload(r: BufferReader): Draft14Message {
602
- const request_id = r.readVarInt();
603
- const error_code = r.readVarInt();
604
- const reason_phrase = r.readString();
605
- return { type: "publish_error", request_id, error_code, reason_phrase };
606
- }
607
-
608
- function decodePublishDonePayload(r: BufferReader): Draft14Message {
609
- const request_id = r.readVarInt();
610
- const status_code = r.readVarInt();
611
- const reason_phrase = r.readString();
612
- return { type: "publish_done", request_id, status_code, reason_phrase };
613
- }
614
-
615
- function decodePublishNamespacePayload(r: BufferReader): Draft14Message {
616
- const request_id = r.readVarInt();
617
- const track_namespace = r.readTuple();
618
- const parameters = decodeParams(r);
619
- return { type: "publish_namespace", request_id, track_namespace, parameters };
620
- }
621
-
622
- function decodePublishNamespaceOkPayload(r: BufferReader): Draft14Message {
623
- const request_id = r.readVarInt();
624
- const parameters = decodeParams(r);
625
- return { type: "publish_namespace_ok", request_id, parameters };
626
- }
627
-
628
- function decodePublishNamespaceErrorPayload(r: BufferReader): Draft14Message {
629
- const request_id = r.readVarInt();
630
- const error_code = r.readVarInt();
631
- const reason_phrase = r.readString();
632
- return { type: "publish_namespace_error", request_id, error_code, reason_phrase };
633
- }
634
-
635
- function decodePublishNamespaceDonePayload(r: BufferReader): Draft14Message {
636
- const request_id = r.readVarInt();
637
- const status_code = r.readVarInt();
638
- const reason_phrase = r.readString();
639
- return { type: "publish_namespace_done", request_id, status_code, reason_phrase };
640
- }
641
-
642
- function decodePublishNamespaceCancelPayload(r: BufferReader): Draft14Message {
643
- const request_id = r.readVarInt();
644
- return { type: "publish_namespace_cancel", request_id };
645
- }
646
-
647
- function decodeSubscribeNamespacePayload(r: BufferReader): Draft14Message {
648
- const request_id = r.readVarInt();
649
- const namespace_prefix = r.readTuple();
650
- const parameters = decodeParams(r);
651
- return { type: "subscribe_namespace", request_id, namespace_prefix, parameters };
652
- }
653
-
654
- function decodeSubscribeNamespaceOkPayload(r: BufferReader): Draft14Message {
655
- const request_id = r.readVarInt();
656
- const parameters = decodeParams(r);
657
- return { type: "subscribe_namespace_ok", request_id, parameters };
658
- }
659
-
660
- function decodeSubscribeNamespaceErrorPayload(r: BufferReader): Draft14Message {
661
- const request_id = r.readVarInt();
662
- const error_code = r.readVarInt();
663
- const reason_phrase = r.readString();
664
- return { type: "subscribe_namespace_error", request_id, error_code, reason_phrase };
665
- }
666
-
667
- function decodeUnsubscribeNamespacePayload(r: BufferReader): Draft14Message {
668
- const request_id = r.readVarInt();
669
- return { type: "unsubscribe_namespace", request_id };
670
- }
671
-
672
- function decodeFetchPayload(r: BufferReader): Draft14Message {
673
- const request_id = r.readVarInt();
674
- const track_namespace = r.readTuple();
675
- const track_name = r.readString();
676
- const start_group = r.readVarInt();
677
- const start_object = r.readVarInt();
678
- const end_group = r.readVarInt();
679
- const parameters = decodeParams(r);
680
- return {
681
- type: "fetch",
682
- request_id,
683
- track_namespace,
684
- track_name,
685
- start_group,
686
- start_object,
687
- end_group,
688
- parameters,
689
- };
690
- }
691
-
692
- function decodeFetchOkPayload(r: BufferReader): Draft14Message {
693
- const request_id = r.readVarInt();
694
- const track_alias = r.readVarInt();
695
- const end_of_track = r.readVarInt();
696
- const parameters = decodeParams(r);
697
- return { type: "fetch_ok", request_id, track_alias, end_of_track, parameters };
698
- }
699
-
700
- function decodeFetchErrorPayload(r: BufferReader): Draft14Message {
701
- const request_id = r.readVarInt();
702
- const error_code = r.readVarInt();
703
- const reason_phrase = r.readString();
704
- return { type: "fetch_error", request_id, error_code, reason_phrase };
705
- }
706
-
707
- function decodeFetchCancelPayload(r: BufferReader): Draft14Message {
708
- const request_id = r.readVarInt();
709
- return { type: "fetch_cancel", request_id };
710
- }
711
-
712
- function decodeTrackStatusPayload(r: BufferReader): Draft14Message {
713
- const request_id = r.readVarInt();
714
- const track_namespace = r.readTuple();
715
- const track_name = r.readString();
716
- const parameters = decodeParams(r);
717
- return { type: "track_status", request_id, track_namespace, track_name, parameters };
718
- }
719
-
720
- function decodeTrackStatusOkPayload(r: BufferReader): Draft14Message {
721
- const request_id = r.readVarInt();
722
- const status_code = r.readVarInt();
723
- const sc = Number(status_code);
724
-
725
- let largest_group: bigint | undefined;
726
- let largest_object: bigint | undefined;
727
-
728
- if (sc === 0 || sc === 3) {
729
- largest_group = r.readVarInt();
730
- largest_object = r.readVarInt();
731
- }
732
-
733
- const parameters = decodeParams(r);
734
-
735
- return {
736
- type: "track_status_ok",
737
- request_id,
738
- status_code,
739
- parameters,
740
- ...(largest_group !== undefined ? { largest_group, largest_object } : {}),
741
- } as Draft14Message;
742
- }
743
-
744
- function decodeTrackStatusErrorPayload(r: BufferReader): Draft14Message {
745
- const request_id = r.readVarInt();
746
- const error_code = r.readVarInt();
747
- const reason_phrase = r.readString();
748
- return { type: "track_status_error", request_id, error_code, reason_phrase };
749
- }
750
-
751
- function decodeGoAwayPayload(r: BufferReader): Draft14Message {
752
- const new_session_uri = r.readString();
753
- return { type: "goaway", new_session_uri };
754
- }
755
-
756
- function decodeMaxRequestIdPayload(r: BufferReader): Draft14Message {
757
- const request_id = r.readVarInt();
758
- return { type: "max_request_id", request_id };
759
- }
760
-
761
- function decodeRequestsBlockedPayload(r: BufferReader): Draft14Message {
762
- const request_id = r.readVarInt();
763
- return { type: "requests_blocked", request_id };
764
- }
765
-
766
- // ─── Payload dispatch tables ───────────────────────────────────────────────────
767
-
768
- const payloadDecoders: ReadonlyMap<bigint, (r: BufferReader) => Draft14Message> = new Map([
769
- [MSG_CLIENT_SETUP, decodeClientSetupPayload],
770
- [MSG_SERVER_SETUP, decodeServerSetupPayload],
771
- [MSG_SUBSCRIBE, decodeSubscribePayload],
772
- [MSG_SUBSCRIBE_OK, decodeSubscribeOkPayload],
773
- [MSG_SUBSCRIBE_UPDATE, decodeSubscribeUpdatePayload],
774
- [MSG_SUBSCRIBE_ERROR, decodeSubscribeErrorPayload],
775
- [MSG_UNSUBSCRIBE, decodeUnsubscribePayload],
776
- [MSG_PUBLISH, decodePublishPayload],
777
- [MSG_PUBLISH_OK, decodePublishOkPayload],
778
- [MSG_PUBLISH_ERROR, decodePublishErrorPayload],
779
- [MSG_PUBLISH_DONE, decodePublishDonePayload],
780
- [MSG_PUBLISH_NAMESPACE, decodePublishNamespacePayload],
781
- [MSG_PUBLISH_NAMESPACE_OK, decodePublishNamespaceOkPayload],
782
- [MSG_PUBLISH_NAMESPACE_ERROR, decodePublishNamespaceErrorPayload],
783
- [MSG_PUBLISH_NAMESPACE_DONE, decodePublishNamespaceDonePayload],
784
- [MSG_PUBLISH_NAMESPACE_CANCEL, decodePublishNamespaceCancelPayload],
785
- [MSG_SUBSCRIBE_NAMESPACE, decodeSubscribeNamespacePayload],
786
- [MSG_SUBSCRIBE_NAMESPACE_OK, decodeSubscribeNamespaceOkPayload],
787
- [MSG_SUBSCRIBE_NAMESPACE_ERROR, decodeSubscribeNamespaceErrorPayload],
788
- [MSG_UNSUBSCRIBE_NAMESPACE, decodeUnsubscribeNamespacePayload],
789
- [MSG_FETCH, decodeFetchPayload],
790
- [MSG_FETCH_OK, decodeFetchOkPayload],
791
- [MSG_FETCH_ERROR, decodeFetchErrorPayload],
792
- [MSG_FETCH_CANCEL, decodeFetchCancelPayload],
793
- [MSG_TRACK_STATUS, decodeTrackStatusPayload],
794
- [MSG_TRACK_STATUS_OK, decodeTrackStatusOkPayload],
795
- [MSG_TRACK_STATUS_ERROR, decodeTrackStatusErrorPayload],
796
- [MSG_GOAWAY, decodeGoAwayPayload],
797
- [MSG_MAX_REQUEST_ID, decodeMaxRequestIdPayload],
798
- [MSG_REQUESTS_BLOCKED, decodeRequestsBlockedPayload],
799
- ]);
800
-
801
- // ─── Public API ────────────────────────────────────────────────────────────────
802
-
803
- /**
804
- * Encode a draft-14 control message with type(varint) + length(uint16 BE) + payload.
805
- */
806
- export function encodeMessage(message: Draft14Message): Uint8Array {
807
- const typeId = MESSAGE_ID_MAP.get(message.type);
808
- if (typeId === undefined) {
809
- throw new Error(`Unknown message type: ${message.type}`);
810
- }
811
-
812
- // Encode payload into a separate buffer
813
- const payloadWriter = new BufferWriter();
814
- encodePayload(message, payloadWriter);
815
- const payload = payloadWriter.finish();
816
-
817
- if (payload.byteLength > 0xffff) {
818
- throw new Error(`Payload too large for 16-bit length: ${payload.byteLength}`);
819
- }
820
-
821
- // Write framed message: type(varint) + length(uint16 BE) + payload
822
- const writer = new BufferWriter();
823
- writer.writeVarInt(typeId);
824
- writer.writeUint8((payload.byteLength >> 8) & 0xff);
825
- writer.writeUint8(payload.byteLength & 0xff);
826
- writer.writeBytes(payload);
827
-
828
- return writer.finish();
829
- }
830
-
831
- function encodePayload(msg: Draft14Message, w: BufferWriter): void {
832
- switch (msg.type) {
833
- case "client_setup":
834
- return encodeClientSetupPayload(msg, w);
835
- case "server_setup":
836
- return encodeServerSetupPayload(msg, w);
837
- case "subscribe":
838
- return encodeSubscribePayload(msg, w);
839
- case "subscribe_ok":
840
- return encodeSubscribeOkPayload(msg, w);
841
- case "subscribe_update":
842
- return encodeSubscribeUpdatePayload(msg, w);
843
- case "subscribe_error":
844
- return encodeSubscribeErrorPayload(msg, w);
845
- case "unsubscribe":
846
- return encodeUnsubscribePayload(msg, w);
847
- case "publish":
848
- return encodePublishPayload(msg, w);
849
- case "publish_ok":
850
- return encodePublishOkPayload(msg, w);
851
- case "publish_error":
852
- return encodePublishErrorPayload(msg, w);
853
- case "publish_done":
854
- return encodePublishDonePayload(msg, w);
855
- case "publish_namespace":
856
- return encodePublishNamespacePayload(msg, w);
857
- case "publish_namespace_ok":
858
- return encodePublishNamespaceOkPayload(msg, w);
859
- case "publish_namespace_error":
860
- return encodePublishNamespaceErrorPayload(msg, w);
861
- case "publish_namespace_done":
862
- return encodePublishNamespaceDonePayload(msg, w);
863
- case "publish_namespace_cancel":
864
- return encodePublishNamespaceCancelPayload(msg, w);
865
- case "subscribe_namespace":
866
- return encodeSubscribeNamespacePayload(msg, w);
867
- case "subscribe_namespace_ok":
868
- return encodeSubscribeNamespaceOkPayload(msg, w);
869
- case "subscribe_namespace_error":
870
- return encodeSubscribeNamespaceErrorPayload(msg, w);
871
- case "unsubscribe_namespace":
872
- return encodeUnsubscribeNamespacePayload(msg, w);
873
- case "fetch":
874
- return encodeFetchPayload(msg, w);
875
- case "fetch_ok":
876
- return encodeFetchOkPayload(msg, w);
877
- case "fetch_error":
878
- return encodeFetchErrorPayload(msg, w);
879
- case "fetch_cancel":
880
- return encodeFetchCancelPayload(msg, w);
881
- case "track_status":
882
- return encodeTrackStatusPayload(msg, w);
883
- case "track_status_ok":
884
- return encodeTrackStatusOkPayload(msg, w);
885
- case "track_status_error":
886
- return encodeTrackStatusErrorPayload(msg, w);
887
- case "goaway":
888
- return encodeGoAwayPayload(msg, w);
889
- case "max_request_id":
890
- return encodeMaxRequestIdPayload(msg, w);
891
- case "requests_blocked":
892
- return encodeRequestsBlockedPayload(msg, w);
893
- default: {
894
- const _exhaustive: never = msg;
895
- throw new Error(`Unhandled message type: ${(_exhaustive as Draft14Message).type}`);
896
- }
897
- }
898
- }
899
-
900
- /**
901
- * Decode a draft-14 control message from bytes (type + uint16 length + payload).
902
- */
903
- export function decodeMessage(bytes: Uint8Array): DecodeResult<Draft14Message> {
904
- try {
905
- const reader = new BufferReader(bytes);
906
- const typeId = reader.readVarInt();
907
-
908
- // Read 16-bit big-endian payload length
909
- const lenHi = reader.readUint8();
910
- const lenLo = reader.readUint8();
911
- const payloadLength = (lenHi << 8) | lenLo;
912
-
913
- // Read exactly payloadLength bytes
914
- const payloadBytes = reader.readBytes(payloadLength);
915
- const payloadReader = new BufferReader(payloadBytes);
916
-
917
- const decoder = payloadDecoders.get(typeId);
918
- if (!decoder) {
919
- return {
920
- ok: false,
921
- error: new DecodeError(
922
- "UNKNOWN_MESSAGE_TYPE",
923
- `Unknown message type ID: 0x${typeId.toString(16)}`,
924
- 0,
925
- ),
926
- };
927
- }
928
-
929
- const message = decoder(payloadReader);
930
- return { ok: true, value: message, bytesRead: reader.offset };
931
- } catch (e) {
932
- if (e instanceof DecodeError) {
933
- return { ok: false, error: e };
934
- }
935
- throw e;
936
- }
937
- }
938
-
939
- // ─── Data Stream Encoding/Decoding ─────────────────────────────────────────────
940
-
941
- /**
942
- * Encode a subgroup stream header + objects.
943
- */
944
- export function encodeSubgroupStream(stream: SubgroupStream): Uint8Array {
945
- const w = new BufferWriter();
946
- w.writeVarInt(0n); // stream type: subgroup
947
- w.writeVarInt(stream.trackAlias);
948
- w.writeVarInt(stream.groupId);
949
- w.writeVarInt(stream.subgroupId);
950
- w.writeUint8(stream.publisherPriority);
951
- for (const obj of stream.objects) {
952
- w.writeVarInt(obj.objectId);
953
- w.writeVarInt(obj.payloadLength);
954
- w.writeBytes(obj.payload);
955
- }
956
- return w.finish();
957
- }
958
-
959
- /**
960
- * Encode a datagram object.
961
- */
962
- export function encodeDatagram(dg: DatagramObject): Uint8Array {
963
- const w = new BufferWriter();
964
- w.writeVarInt(dg.trackAlias);
965
- w.writeVarInt(dg.groupId);
966
- w.writeVarInt(dg.objectId);
967
- w.writeVarInt(dg.payloadLength);
968
- w.writeBytes(dg.payload);
969
- return w.finish();
970
- }
971
-
972
- /**
973
- * Encode a fetch stream header + objects.
974
- */
975
- export function encodeFetchStream(stream: FetchStream): Uint8Array {
976
- const w = new BufferWriter();
977
- w.writeVarInt(2n); // stream type: fetch
978
- w.writeVarInt(stream.subscribeRequestId);
979
- for (const obj of stream.objects) {
980
- w.writeVarInt(obj.objectId);
981
- w.writeVarInt(obj.payloadLength);
982
- w.writeBytes(obj.payload);
983
- }
984
- return w.finish();
985
- }
986
-
987
- /**
988
- * Decode a subgroup data stream from raw bytes.
989
- */
990
- export function decodeSubgroupStream(bytes: Uint8Array): DecodeResult<SubgroupStream> {
991
- try {
992
- const r = new BufferReader(bytes);
993
- const streamType = r.readVarInt();
994
- if (streamType !== 0n) {
995
- return {
996
- ok: false,
997
- error: new DecodeError(
998
- "CONSTRAINT_VIOLATION",
999
- `Expected subgroup stream type 0, got ${streamType}`,
1000
- 0,
1001
- ),
1002
- };
1003
- }
1004
- const trackAlias = r.readVarInt();
1005
- const groupId = r.readVarInt();
1006
- const subgroupId = r.readVarInt();
1007
- const publisherPriority = r.readUint8();
1008
- const objects: ObjectPayload[] = [];
1009
- while (r.remaining > 0) {
1010
- const objectId = r.readVarInt();
1011
- const payloadLength = Number(r.readVarInt());
1012
- const payload = r.readBytes(payloadLength);
1013
- objects.push({ type: "object", objectId, payloadLength, payload });
1014
- }
1015
- return {
1016
- ok: true,
1017
- value: { type: "subgroup", trackAlias, groupId, subgroupId, publisherPriority, objects },
1018
- bytesRead: r.offset,
1019
- };
1020
- } catch (e) {
1021
- if (e instanceof DecodeError) return { ok: false, error: e };
1022
- throw e;
1023
- }
1024
- }
1025
-
1026
- /**
1027
- * Decode a datagram object from raw bytes.
1028
- */
1029
- export function decodeDatagram(bytes: Uint8Array): DecodeResult<DatagramObject> {
1030
- try {
1031
- const r = new BufferReader(bytes);
1032
- const trackAlias = r.readVarInt();
1033
- const groupId = r.readVarInt();
1034
- const objectId = r.readVarInt();
1035
- const payloadLength = Number(r.readVarInt());
1036
- const payload = r.readBytes(payloadLength);
1037
- return {
1038
- ok: true,
1039
- value: { type: "datagram", trackAlias, groupId, objectId, payloadLength, payload },
1040
- bytesRead: r.offset,
1041
- };
1042
- } catch (e) {
1043
- if (e instanceof DecodeError) return { ok: false, error: e };
1044
- throw e;
1045
- }
1046
- }
1047
-
1048
- /**
1049
- * Decode a fetch data stream from raw bytes.
1050
- */
1051
- export function decodeFetchStream(bytes: Uint8Array): DecodeResult<FetchStream> {
1052
- try {
1053
- const r = new BufferReader(bytes);
1054
- const streamType = r.readVarInt();
1055
- if (streamType !== 2n) {
1056
- return {
1057
- ok: false,
1058
- error: new DecodeError(
1059
- "CONSTRAINT_VIOLATION",
1060
- `Expected fetch stream type 2, got ${streamType}`,
1061
- 0,
1062
- ),
1063
- };
1064
- }
1065
- const subscribeRequestId = r.readVarInt();
1066
- const objects: ObjectPayload[] = [];
1067
- while (r.remaining > 0) {
1068
- const objectId = r.readVarInt();
1069
- const payloadLength = Number(r.readVarInt());
1070
- const payload = r.readBytes(payloadLength);
1071
- objects.push({ type: "object", objectId, payloadLength, payload });
1072
- }
1073
- return {
1074
- ok: true,
1075
- value: { type: "fetch", subscribeRequestId, objects },
1076
- bytesRead: r.offset,
1077
- };
1078
- } catch (e) {
1079
- if (e instanceof DecodeError) return { ok: false, error: e };
1080
- throw e;
1081
- }
1082
- }
1083
-
1084
- /**
1085
- * Decode a data stream, dispatching by stream type.
1086
- */
1087
- export function decodeDataStream(
1088
- streamType: "subgroup" | "datagram" | "fetch",
1089
- bytes: Uint8Array,
1090
- ): DecodeResult<Draft14DataStream> {
1091
- switch (streamType) {
1092
- case "subgroup":
1093
- return decodeSubgroupStream(bytes);
1094
- case "datagram":
1095
- return decodeDatagram(bytes);
1096
- case "fetch":
1097
- return decodeFetchStream(bytes);
1098
- default: {
1099
- const _exhaustive: never = streamType;
1100
- throw new Error(`Unknown stream type: ${_exhaustive}`);
1101
- }
1102
- }
1103
- }
1104
-
1105
- // ─── Stream Decoders ───────────────────────────────────────────────────────────
1106
-
1107
- /**
1108
- * Create a TransformStream that decodes a continuous byte stream (e.g. the
1109
- * WebTransport bidirectional control stream) into individual Draft14Message
1110
- * objects. Uses the varint(type) + uint16_BE(length) + payload framing.
1111
- */
1112
- export function createStreamDecoder(): TransformStream<Uint8Array, Draft14Message> {
1113
- let buffer = new Uint8Array(0);
1114
-
1115
- return new TransformStream<Uint8Array, Draft14Message>({
1116
- transform(chunk, controller) {
1117
- // Accumulate incoming data
1118
- const newBuffer = new Uint8Array(buffer.length + chunk.length);
1119
- newBuffer.set(buffer, 0);
1120
- newBuffer.set(chunk, buffer.length);
1121
- buffer = newBuffer;
1122
-
1123
- // Try to decode messages from the buffer
1124
- while (buffer.length > 0) {
1125
- const result = decodeMessage(buffer);
1126
- if (!result.ok) {
1127
- if (result.error.code === "UNEXPECTED_END") {
1128
- // Need more data -- wait for next chunk
1129
- break;
1130
- }
1131
- // Fatal decode error
1132
- controller.error(result.error);
1133
- return;
1134
- }
1135
- controller.enqueue(result.value);
1136
- // Advance the buffer past the consumed bytes
1137
- buffer = buffer.slice(result.bytesRead);
1138
- }
1139
- },
1140
-
1141
- flush(controller) {
1142
- // If there is remaining data in the buffer, it is a truncated message
1143
- if (buffer.length > 0) {
1144
- controller.error(
1145
- new DecodeError("UNEXPECTED_END", "Stream ended with incomplete message data", 0),
1146
- );
1147
- }
1148
- },
1149
- });
1150
- }
1151
-
1152
- /**
1153
- * Create a TransformStream that decodes a subgroup data stream.
1154
- * First emits a SubgroupStreamHeader, then emits ObjectPayload events.
1155
- *
1156
- * Subgroup stream framing:
1157
- * varint(0x00) + varint(trackAlias) + varint(groupId) + varint(subgroupId)
1158
- * + uint8(publisherPriority) + [varint(objectId) + varint(payloadLength) + bytes(payload)]*
1159
- */
1160
- export function createSubgroupStreamDecoder(): TransformStream<
1161
- Uint8Array,
1162
- SubgroupStreamHeader | ObjectPayload
1163
- > {
1164
- let buffer = new Uint8Array(0);
1165
- let headerEmitted = false;
1166
-
1167
- return new TransformStream<Uint8Array, SubgroupStreamHeader | ObjectPayload>({
1168
- transform(chunk, controller) {
1169
- const newBuffer = new Uint8Array(buffer.length + chunk.length);
1170
- newBuffer.set(buffer, 0);
1171
- newBuffer.set(chunk, buffer.length);
1172
- buffer = newBuffer;
1173
-
1174
- // First, parse the header if not yet emitted
1175
- if (!headerEmitted) {
1176
- try {
1177
- const r = new BufferReader(buffer);
1178
- const streamType = r.readVarInt();
1179
- if (streamType !== 0n) {
1180
- controller.error(
1181
- new DecodeError(
1182
- "CONSTRAINT_VIOLATION",
1183
- `Expected subgroup stream type 0, got ${streamType}`,
1184
- 0,
1185
- ),
1186
- );
1187
- return;
1188
- }
1189
- const trackAlias = r.readVarInt();
1190
- const groupId = r.readVarInt();
1191
- const subgroupId = r.readVarInt();
1192
- const publisherPriority = r.readUint8();
1193
-
1194
- controller.enqueue({
1195
- type: "subgroup_header",
1196
- trackAlias,
1197
- groupId,
1198
- subgroupId,
1199
- publisherPriority,
1200
- });
1201
- headerEmitted = true;
1202
- buffer = buffer.slice(r.offset);
1203
- } catch (e) {
1204
- if (e instanceof DecodeError && e.code === "UNEXPECTED_END") {
1205
- return; // Need more data
1206
- }
1207
- controller.error(e);
1208
- return;
1209
- }
1210
- }
1211
-
1212
- // Parse objects
1213
- while (buffer.length > 0) {
1214
- try {
1215
- const r = new BufferReader(buffer);
1216
- const objectId = r.readVarInt();
1217
- const payloadLength = Number(r.readVarInt());
1218
- const payload = r.readBytes(payloadLength);
1219
- controller.enqueue({ type: "object", objectId, payloadLength, payload });
1220
- buffer = buffer.slice(r.offset);
1221
- } catch (e) {
1222
- if (e instanceof DecodeError && e.code === "UNEXPECTED_END") {
1223
- break; // Need more data
1224
- }
1225
- controller.error(e);
1226
- return;
1227
- }
1228
- }
1229
- },
1230
-
1231
- flush(controller) {
1232
- if (buffer.length > 0) {
1233
- controller.error(new DecodeError("UNEXPECTED_END", "Stream ended with incomplete data", 0));
1234
- }
1235
- },
1236
- });
1237
- }
1238
-
1239
- /**
1240
- * Create a TransformStream that decodes a fetch data stream.
1241
- * First emits a FetchStreamHeader, then emits ObjectPayload events.
1242
- *
1243
- * Fetch stream framing:
1244
- * varint(0x02) + varint(subscribeRequestId)
1245
- * + [varint(objectId) + varint(payloadLength) + bytes(payload)]*
1246
- */
1247
- export function createFetchStreamDecoder(): TransformStream<
1248
- Uint8Array,
1249
- FetchStreamHeader | ObjectPayload
1250
- > {
1251
- let buffer = new Uint8Array(0);
1252
- let headerEmitted = false;
1253
-
1254
- return new TransformStream<Uint8Array, FetchStreamHeader | ObjectPayload>({
1255
- transform(chunk, controller) {
1256
- const newBuffer = new Uint8Array(buffer.length + chunk.length);
1257
- newBuffer.set(buffer, 0);
1258
- newBuffer.set(chunk, buffer.length);
1259
- buffer = newBuffer;
1260
-
1261
- // First, parse the header if not yet emitted
1262
- if (!headerEmitted) {
1263
- try {
1264
- const r = new BufferReader(buffer);
1265
- const streamType = r.readVarInt();
1266
- if (streamType !== 2n) {
1267
- controller.error(
1268
- new DecodeError(
1269
- "CONSTRAINT_VIOLATION",
1270
- `Expected fetch stream type 2, got ${streamType}`,
1271
- 0,
1272
- ),
1273
- );
1274
- return;
1275
- }
1276
- const subscribeRequestId = r.readVarInt();
1277
-
1278
- controller.enqueue({
1279
- type: "fetch_header",
1280
- subscribeRequestId,
1281
- });
1282
- headerEmitted = true;
1283
- buffer = buffer.slice(r.offset);
1284
- } catch (e) {
1285
- if (e instanceof DecodeError && e.code === "UNEXPECTED_END") {
1286
- return; // Need more data
1287
- }
1288
- controller.error(e);
1289
- return;
1290
- }
1291
- }
1292
-
1293
- // Parse objects
1294
- while (buffer.length > 0) {
1295
- try {
1296
- const r = new BufferReader(buffer);
1297
- const objectId = r.readVarInt();
1298
- const payloadLength = Number(r.readVarInt());
1299
- const payload = r.readBytes(payloadLength);
1300
- controller.enqueue({ type: "object", objectId, payloadLength, payload });
1301
- buffer = buffer.slice(r.offset);
1302
- } catch (e) {
1303
- if (e instanceof DecodeError && e.code === "UNEXPECTED_END") {
1304
- break; // Need more data
1305
- }
1306
- controller.error(e);
1307
- return;
1308
- }
1309
- }
1310
- },
1311
-
1312
- flush(controller) {
1313
- if (buffer.length > 0) {
1314
- controller.error(new DecodeError("UNEXPECTED_END", "Stream ended with incomplete data", 0));
1315
- }
1316
- },
1317
- });
1318
- }
1319
-
1320
- /**
1321
- * Create a unified auto-detecting data stream decoder.
1322
- * Reads the stream type varint from the first bytes, then delegates to
1323
- * the appropriate decoder (subgroup or fetch).
1324
- */
1325
- export function createDataStreamDecoder(): TransformStream<Uint8Array, DataStreamEvent> {
1326
- let buffer = new Uint8Array(0);
1327
- let detectedType: "subgroup" | "fetch" | null = null;
1328
- let headerEmitted = false;
1329
-
1330
- // We peek the stream type then create the appropriate inner decoder,
1331
- // forwarding all remaining data through it. To avoid complexity of
1332
- // piping inner TransformStreams we just inline the same logic.
1333
-
1334
- return new TransformStream<Uint8Array, DataStreamEvent>({
1335
- transform(chunk, controller) {
1336
- const newBuffer = new Uint8Array(buffer.length + chunk.length);
1337
- newBuffer.set(buffer, 0);
1338
- newBuffer.set(chunk, buffer.length);
1339
- buffer = newBuffer;
1340
-
1341
- // Detect stream type if not yet done
1342
- if (detectedType === null) {
1343
- try {
1344
- const r = new BufferReader(buffer);
1345
- const streamType = r.readVarInt();
1346
- if (streamType === 0n) {
1347
- detectedType = "subgroup";
1348
- } else if (streamType === 2n) {
1349
- detectedType = "fetch";
1350
- } else {
1351
- controller.error(
1352
- new DecodeError("CONSTRAINT_VIOLATION", `Unknown data stream type: ${streamType}`, 0),
1353
- );
1354
- return;
1355
- }
1356
- // Do NOT advance buffer - the inner header parser expects the stream type
1357
- } catch (e) {
1358
- if (e instanceof DecodeError && e.code === "UNEXPECTED_END") {
1359
- return; // Need more data
1360
- }
1361
- controller.error(e);
1362
- return;
1363
- }
1364
- }
1365
-
1366
- if (detectedType === "subgroup") {
1367
- // Parse subgroup header if not yet emitted
1368
- if (!headerEmitted) {
1369
- try {
1370
- const r = new BufferReader(buffer);
1371
- r.readVarInt(); // stream type (0x00)
1372
- const trackAlias = r.readVarInt();
1373
- const groupId = r.readVarInt();
1374
- const subgroupId = r.readVarInt();
1375
- const publisherPriority = r.readUint8();
1376
- controller.enqueue({
1377
- type: "subgroup_header",
1378
- trackAlias,
1379
- groupId,
1380
- subgroupId,
1381
- publisherPriority,
1382
- });
1383
- headerEmitted = true;
1384
- buffer = buffer.slice(r.offset);
1385
- } catch (e) {
1386
- if (e instanceof DecodeError && e.code === "UNEXPECTED_END") {
1387
- return;
1388
- }
1389
- controller.error(e);
1390
- return;
1391
- }
1392
- }
1393
- } else {
1394
- // Parse fetch header if not yet emitted
1395
- if (!headerEmitted) {
1396
- try {
1397
- const r = new BufferReader(buffer);
1398
- r.readVarInt(); // stream type (0x02)
1399
- const subscribeRequestId = r.readVarInt();
1400
- controller.enqueue({
1401
- type: "fetch_header",
1402
- subscribeRequestId,
1403
- });
1404
- headerEmitted = true;
1405
- buffer = buffer.slice(r.offset);
1406
- } catch (e) {
1407
- if (e instanceof DecodeError && e.code === "UNEXPECTED_END") {
1408
- return;
1409
- }
1410
- controller.error(e);
1411
- return;
1412
- }
1413
- }
1414
- }
1415
-
1416
- // Parse objects
1417
- while (buffer.length > 0) {
1418
- try {
1419
- const r = new BufferReader(buffer);
1420
- const objectId = r.readVarInt();
1421
- const payloadLength = Number(r.readVarInt());
1422
- const payload = r.readBytes(payloadLength);
1423
- controller.enqueue({ type: "object", objectId, payloadLength, payload });
1424
- buffer = buffer.slice(r.offset);
1425
- } catch (e) {
1426
- if (e instanceof DecodeError && e.code === "UNEXPECTED_END") {
1427
- break;
1428
- }
1429
- controller.error(e);
1430
- return;
1431
- }
1432
- }
1433
- },
1434
-
1435
- flush(controller) {
1436
- if (buffer.length > 0) {
1437
- controller.error(new DecodeError("UNEXPECTED_END", "Stream ended with incomplete data", 0));
1438
- }
1439
- },
1440
- });
1441
- }
1442
-
1443
- // ─── Codec Factory ─────────────────────────────────────────────────────────────
1444
-
1445
- export interface Draft14Codec extends BaseCodec<Draft14Message> {
1446
- readonly draft: "draft-ietf-moq-transport-14";
1447
- encodeSubgroupStream(stream: SubgroupStream): Uint8Array;
1448
- encodeDatagram(dg: DatagramObject): Uint8Array;
1449
- encodeFetchStream(stream: FetchStream): Uint8Array;
1450
- decodeSubgroupStream(bytes: Uint8Array): DecodeResult<SubgroupStream>;
1451
- decodeDatagram(bytes: Uint8Array): DecodeResult<DatagramObject>;
1452
- decodeFetchStream(bytes: Uint8Array): DecodeResult<FetchStream>;
1453
- decodeDataStream(
1454
- streamType: "subgroup" | "datagram" | "fetch",
1455
- bytes: Uint8Array,
1456
- ): DecodeResult<Draft14DataStream>;
1457
- createStreamDecoder(): TransformStream<Uint8Array, Draft14Message>;
1458
- createSubgroupStreamDecoder(): TransformStream<Uint8Array, SubgroupStreamHeader | ObjectPayload>;
1459
- createFetchStreamDecoder(): TransformStream<Uint8Array, FetchStreamHeader | ObjectPayload>;
1460
- createDataStreamDecoder(): TransformStream<Uint8Array, DataStreamEvent>;
1461
- }
1462
-
1463
- export function createDraft14Codec(): Draft14Codec {
1464
- return {
1465
- draft: "draft-ietf-moq-transport-14",
1466
- encodeMessage,
1467
- decodeMessage,
1468
- encodeSubgroupStream,
1469
- encodeDatagram,
1470
- encodeFetchStream,
1471
- decodeSubgroupStream,
1472
- decodeDatagram,
1473
- decodeFetchStream,
1474
- decodeDataStream,
1475
- createStreamDecoder,
1476
- createSubgroupStreamDecoder,
1477
- createFetchStreamDecoder,
1478
- createDataStreamDecoder,
1479
- };
1480
- }
1
+ import { bytesToHex, hexToBytes } from "../../core/hex.js";
2
+ import { BufferReader } from "../../core/buffer-reader.js";
3
+ import { BufferWriter } from "../../core/buffer-writer.js";
4
+ import type { BaseCodec, DecodeResult } from "../../core/types.js";
5
+ import { DecodeError } from "../../core/types.js";
6
+ import {
7
+ MESSAGE_ID_MAP,
8
+ MSG_CLIENT_SETUP,
9
+ MSG_FETCH,
10
+ MSG_FETCH_CANCEL,
11
+ MSG_FETCH_ERROR,
12
+ MSG_FETCH_OK,
13
+ MSG_GOAWAY,
14
+ MSG_MAX_REQUEST_ID,
15
+ MSG_PUBLISH,
16
+ MSG_PUBLISH_DONE,
17
+ MSG_PUBLISH_ERROR,
18
+ MSG_PUBLISH_NAMESPACE,
19
+ MSG_PUBLISH_NAMESPACE_CANCEL,
20
+ MSG_PUBLISH_NAMESPACE_DONE,
21
+ MSG_PUBLISH_NAMESPACE_ERROR,
22
+ MSG_PUBLISH_NAMESPACE_OK,
23
+ MSG_PUBLISH_OK,
24
+ MSG_REQUESTS_BLOCKED,
25
+ MSG_SERVER_SETUP,
26
+ MSG_SUBSCRIBE,
27
+ MSG_SUBSCRIBE_ERROR,
28
+ MSG_SUBSCRIBE_NAMESPACE,
29
+ MSG_SUBSCRIBE_NAMESPACE_ERROR,
30
+ MSG_SUBSCRIBE_NAMESPACE_OK,
31
+ MSG_SUBSCRIBE_OK,
32
+ MSG_SUBSCRIBE_UPDATE,
33
+ MSG_TRACK_STATUS,
34
+ MSG_TRACK_STATUS_ERROR,
35
+ MSG_TRACK_STATUS_OK,
36
+ MSG_UNSUBSCRIBE,
37
+ MSG_UNSUBSCRIBE_NAMESPACE,
38
+ PARAM_MAX_REQUEST_ID,
39
+ PARAM_PATH,
40
+ PARAM_ROLE,
41
+ } from "./messages.js";
42
+ import type {
43
+ DatagramObject,
44
+ DataStreamEvent,
45
+ Draft14DataStream,
46
+ Draft14Message,
47
+ Draft14Params,
48
+ FetchObjectPayload,
49
+ FetchStream,
50
+ FetchStreamHeader,
51
+ ObjectPayload,
52
+ SubgroupStream,
53
+ SubgroupStreamHeader,
54
+ UnknownParam,
55
+ } from "./types.js";
56
+ // ─── Parameter Encoding/Decoding ───────────────────────────────────────────────
57
+
58
+ function encodeParams(params: Draft14Params, writer: BufferWriter): void {
59
+ // Count total params
60
+ let count = 0;
61
+ if (params.role !== undefined) count++;
62
+ if (params.path !== undefined) count++;
63
+ if (params.max_request_id !== undefined) count++;
64
+ if (params.unknown) count += params.unknown.length;
65
+
66
+ writer.writeVarInt(count);
67
+
68
+ // ROLE (0x00) - even, varint value
69
+ if (params.role !== undefined) {
70
+ writer.writeVarInt(PARAM_ROLE);
71
+ writer.writeVarInt(params.role);
72
+ }
73
+
74
+ // PATH (0x01) - odd, length-prefixed bytes
75
+ if (params.path !== undefined) {
76
+ writer.writeVarInt(PARAM_PATH);
77
+ const encoded = new TextEncoder().encode(params.path);
78
+ writer.writeVarInt(encoded.byteLength);
79
+ writer.writeBytes(encoded);
80
+ }
81
+
82
+ // MAX_REQUEST_ID (0x02) - even, varint value
83
+ if (params.max_request_id !== undefined) {
84
+ writer.writeVarInt(PARAM_MAX_REQUEST_ID);
85
+ writer.writeVarInt(params.max_request_id);
86
+ }
87
+
88
+ // Unknown params
89
+ if (params.unknown) {
90
+ for (const u of params.unknown) {
91
+ const id = BigInt(u.id);
92
+ writer.writeVarInt(id);
93
+ const raw = hexToBytes(u.raw_hex);
94
+ writer.writeVarInt(raw.byteLength);
95
+ writer.writeBytes(raw);
96
+ }
97
+ }
98
+ }
99
+
100
+ function decodeParams(reader: BufferReader): Draft14Params {
101
+ const count = Number(reader.readVarInt());
102
+ const result: Draft14Params = {};
103
+ const unknown: UnknownParam[] = [];
104
+
105
+ for (let i = 0; i < count; i++) {
106
+ const paramType = reader.readVarInt();
107
+
108
+ if (paramType % 2n === 0n) {
109
+ // Even: value is a varint directly
110
+ const value = reader.readVarInt();
111
+ if (paramType === PARAM_ROLE) {
112
+ result.role = value;
113
+ } else if (paramType === PARAM_MAX_REQUEST_ID) {
114
+ result.max_request_id = value;
115
+ } else {
116
+ // Unknown even param — encode the varint value as raw bytes
117
+ const tmpWriter = new BufferWriter(16);
118
+ tmpWriter.writeVarInt(value);
119
+ const raw = tmpWriter.finish();
120
+ unknown.push({
121
+ id: `0x${paramType.toString(16)}`,
122
+ length: raw.byteLength,
123
+ raw_hex: bytesToHex(raw),
124
+ });
125
+ }
126
+ } else {
127
+ // Odd: value is length-prefixed bytes
128
+ const length = Number(reader.readVarInt());
129
+ const bytes = reader.readBytes(length);
130
+ if (paramType === PARAM_PATH) {
131
+ result.path = new TextDecoder().decode(bytes);
132
+ } else {
133
+ unknown.push({
134
+ id: `0x${paramType.toString(16)}`,
135
+ length,
136
+ raw_hex: bytesToHex(bytes),
137
+ });
138
+ }
139
+ }
140
+ }
141
+
142
+ if (unknown.length > 0) {
143
+ result.unknown = unknown;
144
+ }
145
+
146
+ return result;
147
+ }
148
+
149
+ // ─── Payload Encoders ──────────────────────────────────────────────────────────
150
+
151
+ function encodeClientSetupPayload(
152
+ msg: Draft14Message & { type: "client_setup" },
153
+ w: BufferWriter,
154
+ ): void {
155
+ w.writeVarInt(msg.supported_versions.length);
156
+ for (const v of msg.supported_versions) {
157
+ w.writeVarInt(v);
158
+ }
159
+ encodeParams(msg.parameters, w);
160
+ }
161
+
162
+ function encodeServerSetupPayload(
163
+ msg: Draft14Message & { type: "server_setup" },
164
+ w: BufferWriter,
165
+ ): void {
166
+ w.writeVarInt(msg.selected_version);
167
+ encodeParams(msg.parameters, w);
168
+ }
169
+
170
+ function encodeSubscribePayload(
171
+ msg: Draft14Message & { type: "subscribe" },
172
+ w: BufferWriter,
173
+ ): void {
174
+ w.writeVarInt(msg.request_id);
175
+ w.writeTuple(msg.track_namespace);
176
+ w.writeString(msg.track_name);
177
+ w.writeUint8(Number(msg.subscriber_priority));
178
+ w.writeVarInt(msg.group_order);
179
+ w.writeVarInt(msg.forward);
180
+ w.writeVarInt(msg.filter_type);
181
+ const ft = Number(msg.filter_type);
182
+ if (ft >= 3) {
183
+ w.writeVarInt(msg.start_group!);
184
+ w.writeVarInt(msg.start_object!);
185
+ }
186
+ if (ft === 4) {
187
+ w.writeVarInt(msg.end_group!);
188
+ }
189
+ encodeParams(msg.parameters, w);
190
+ }
191
+
192
+ function encodeSubscribeOkPayload(
193
+ msg: Draft14Message & { type: "subscribe_ok" },
194
+ w: BufferWriter,
195
+ ): void {
196
+ w.writeVarInt(msg.request_id);
197
+ w.writeVarInt(msg.track_alias);
198
+ w.writeVarInt(msg.expires);
199
+ w.writeVarInt(msg.group_order);
200
+ w.writeVarInt(msg.content_exists);
201
+ if (Number(msg.content_exists) === 1) {
202
+ w.writeVarInt(msg.largest_group!);
203
+ w.writeVarInt(msg.largest_object!);
204
+ }
205
+ encodeParams(msg.parameters, w);
206
+ }
207
+
208
+ function encodeSubscribeUpdatePayload(
209
+ msg: Draft14Message & { type: "subscribe_update" },
210
+ w: BufferWriter,
211
+ ): void {
212
+ w.writeVarInt(msg.request_id);
213
+ w.writeVarInt(msg.start_group);
214
+ w.writeVarInt(msg.start_object);
215
+ w.writeVarInt(msg.end_group);
216
+ w.writeUint8(Number(msg.subscriber_priority));
217
+ w.writeVarInt(msg.forward);
218
+ encodeParams(msg.parameters, w);
219
+ }
220
+
221
+ function encodeSubscribeErrorPayload(
222
+ msg: Draft14Message & { type: "subscribe_error" },
223
+ w: BufferWriter,
224
+ ): void {
225
+ w.writeVarInt(msg.request_id);
226
+ w.writeVarInt(msg.error_code);
227
+ w.writeString(msg.reason_phrase);
228
+ }
229
+
230
+ function encodeUnsubscribePayload(
231
+ msg: Draft14Message & { type: "unsubscribe" },
232
+ w: BufferWriter,
233
+ ): void {
234
+ w.writeVarInt(msg.request_id);
235
+ }
236
+
237
+ function encodePublishPayload(msg: Draft14Message & { type: "publish" }, w: BufferWriter): void {
238
+ w.writeVarInt(msg.request_id);
239
+ w.writeTuple(msg.track_namespace);
240
+ w.writeString(msg.track_name);
241
+ w.writeVarInt(msg.forward);
242
+ encodeParams(msg.parameters, w);
243
+ }
244
+
245
+ function encodePublishOkPayload(
246
+ msg: Draft14Message & { type: "publish_ok" },
247
+ w: BufferWriter,
248
+ ): void {
249
+ w.writeVarInt(msg.request_id);
250
+ w.writeVarInt(msg.track_alias);
251
+ w.writeVarInt(msg.forward);
252
+ encodeParams(msg.parameters, w);
253
+ }
254
+
255
+ function encodePublishErrorPayload(
256
+ msg: Draft14Message & { type: "publish_error" },
257
+ w: BufferWriter,
258
+ ): void {
259
+ w.writeVarInt(msg.request_id);
260
+ w.writeVarInt(msg.error_code);
261
+ w.writeString(msg.reason_phrase);
262
+ }
263
+
264
+ function encodePublishDonePayload(
265
+ msg: Draft14Message & { type: "publish_done" },
266
+ w: BufferWriter,
267
+ ): void {
268
+ w.writeVarInt(msg.request_id);
269
+ w.writeVarInt(msg.status_code);
270
+ w.writeString(msg.reason_phrase);
271
+ }
272
+
273
+ function encodePublishNamespacePayload(
274
+ msg: Draft14Message & { type: "publish_namespace" },
275
+ w: BufferWriter,
276
+ ): void {
277
+ w.writeVarInt(msg.request_id);
278
+ w.writeTuple(msg.track_namespace);
279
+ encodeParams(msg.parameters, w);
280
+ }
281
+
282
+ function encodePublishNamespaceOkPayload(
283
+ msg: Draft14Message & { type: "publish_namespace_ok" },
284
+ w: BufferWriter,
285
+ ): void {
286
+ w.writeVarInt(msg.request_id);
287
+ encodeParams(msg.parameters, w);
288
+ }
289
+
290
+ function encodePublishNamespaceErrorPayload(
291
+ msg: Draft14Message & { type: "publish_namespace_error" },
292
+ w: BufferWriter,
293
+ ): void {
294
+ w.writeVarInt(msg.request_id);
295
+ w.writeVarInt(msg.error_code);
296
+ w.writeString(msg.reason_phrase);
297
+ }
298
+
299
+ function encodePublishNamespaceDonePayload(
300
+ msg: Draft14Message & { type: "publish_namespace_done" },
301
+ w: BufferWriter,
302
+ ): void {
303
+ w.writeVarInt(msg.request_id);
304
+ w.writeVarInt(msg.status_code);
305
+ w.writeString(msg.reason_phrase);
306
+ }
307
+
308
+ function encodePublishNamespaceCancelPayload(
309
+ msg: Draft14Message & { type: "publish_namespace_cancel" },
310
+ w: BufferWriter,
311
+ ): void {
312
+ w.writeVarInt(msg.request_id);
313
+ }
314
+
315
+ function encodeSubscribeNamespacePayload(
316
+ msg: Draft14Message & { type: "subscribe_namespace" },
317
+ w: BufferWriter,
318
+ ): void {
319
+ w.writeVarInt(msg.request_id);
320
+ w.writeTuple(msg.namespace_prefix);
321
+ encodeParams(msg.parameters, w);
322
+ }
323
+
324
+ function encodeSubscribeNamespaceOkPayload(
325
+ msg: Draft14Message & { type: "subscribe_namespace_ok" },
326
+ w: BufferWriter,
327
+ ): void {
328
+ w.writeVarInt(msg.request_id);
329
+ encodeParams(msg.parameters, w);
330
+ }
331
+
332
+ function encodeSubscribeNamespaceErrorPayload(
333
+ msg: Draft14Message & { type: "subscribe_namespace_error" },
334
+ w: BufferWriter,
335
+ ): void {
336
+ w.writeVarInt(msg.request_id);
337
+ w.writeVarInt(msg.error_code);
338
+ w.writeString(msg.reason_phrase);
339
+ }
340
+
341
+ function encodeUnsubscribeNamespacePayload(
342
+ msg: Draft14Message & { type: "unsubscribe_namespace" },
343
+ w: BufferWriter,
344
+ ): void {
345
+ w.writeVarInt(msg.request_id);
346
+ }
347
+
348
+ function encodeFetchPayload(msg: Draft14Message & { type: "fetch" }, w: BufferWriter): void {
349
+ w.writeVarInt(msg.request_id);
350
+ w.writeTuple(msg.track_namespace);
351
+ w.writeString(msg.track_name);
352
+ w.writeVarInt(msg.start_group);
353
+ w.writeVarInt(msg.start_object);
354
+ w.writeVarInt(msg.end_group);
355
+ encodeParams(msg.parameters, w);
356
+ }
357
+
358
+ function encodeFetchOkPayload(msg: Draft14Message & { type: "fetch_ok" }, w: BufferWriter): void {
359
+ w.writeVarInt(msg.request_id);
360
+ w.writeVarInt(msg.track_alias);
361
+ w.writeVarInt(msg.end_of_track);
362
+ encodeParams(msg.parameters, w);
363
+ }
364
+
365
+ function encodeFetchErrorPayload(
366
+ msg: Draft14Message & { type: "fetch_error" },
367
+ w: BufferWriter,
368
+ ): void {
369
+ w.writeVarInt(msg.request_id);
370
+ w.writeVarInt(msg.error_code);
371
+ w.writeString(msg.reason_phrase);
372
+ }
373
+
374
+ function encodeFetchCancelPayload(
375
+ msg: Draft14Message & { type: "fetch_cancel" },
376
+ w: BufferWriter,
377
+ ): void {
378
+ w.writeVarInt(msg.request_id);
379
+ }
380
+
381
+ function encodeTrackStatusPayload(
382
+ msg: Draft14Message & { type: "track_status" },
383
+ w: BufferWriter,
384
+ ): void {
385
+ w.writeVarInt(msg.request_id);
386
+ w.writeTuple(msg.track_namespace);
387
+ w.writeString(msg.track_name);
388
+ encodeParams(msg.parameters, w);
389
+ }
390
+
391
+ function encodeTrackStatusOkPayload(
392
+ msg: Draft14Message & { type: "track_status_ok" },
393
+ w: BufferWriter,
394
+ ): void {
395
+ w.writeVarInt(msg.request_id);
396
+ w.writeVarInt(msg.status_code);
397
+ const sc = Number(msg.status_code);
398
+ if (sc === 0 || sc === 3) {
399
+ w.writeVarInt(msg.largest_group!);
400
+ w.writeVarInt(msg.largest_object!);
401
+ }
402
+ encodeParams(msg.parameters, w);
403
+ }
404
+
405
+ function encodeTrackStatusErrorPayload(
406
+ msg: Draft14Message & { type: "track_status_error" },
407
+ w: BufferWriter,
408
+ ): void {
409
+ w.writeVarInt(msg.request_id);
410
+ w.writeVarInt(msg.error_code);
411
+ w.writeString(msg.reason_phrase);
412
+ }
413
+
414
+ function encodeGoAwayPayload(msg: Draft14Message & { type: "goaway" }, w: BufferWriter): void {
415
+ w.writeString(msg.new_session_uri);
416
+ }
417
+
418
+ function encodeMaxRequestIdPayload(
419
+ msg: Draft14Message & { type: "max_request_id" },
420
+ w: BufferWriter,
421
+ ): void {
422
+ w.writeVarInt(msg.request_id);
423
+ }
424
+
425
+ function encodeRequestsBlockedPayload(
426
+ msg: Draft14Message & { type: "requests_blocked" },
427
+ w: BufferWriter,
428
+ ): void {
429
+ w.writeVarInt(msg.request_id);
430
+ }
431
+
432
+ // ─── Payload Decoders ──────────────────────────────────────────────────────────
433
+
434
+ function decodeClientSetupPayload(r: BufferReader): Draft14Message {
435
+ const numVersions = Number(r.readVarInt());
436
+ if (numVersions === 0) {
437
+ throw new DecodeError(
438
+ "CONSTRAINT_VIOLATION",
439
+ "CLIENT_SETUP must offer at least one version",
440
+ r.offset,
441
+ );
442
+ }
443
+ const supported_versions: bigint[] = [];
444
+ for (let i = 0; i < numVersions; i++) {
445
+ supported_versions.push(r.readVarInt());
446
+ }
447
+ const parameters = decodeParams(r);
448
+ return { type: "client_setup", supported_versions, parameters };
449
+ }
450
+
451
+ function decodeServerSetupPayload(r: BufferReader): Draft14Message {
452
+ const selected_version = r.readVarInt();
453
+ const parameters = decodeParams(r);
454
+ return { type: "server_setup", selected_version, parameters };
455
+ }
456
+
457
+ function decodeSubscribePayload(r: BufferReader): Draft14Message {
458
+ const request_id = r.readVarInt();
459
+ const track_namespace = r.readTuple();
460
+ const track_name = r.readString();
461
+ const subscriber_priority = BigInt(r.readUint8());
462
+ const group_order = r.readVarInt();
463
+ const forward = r.readVarInt();
464
+ const filter_type = r.readVarInt();
465
+ const ft = Number(filter_type);
466
+
467
+ if (ft < 1 || ft > 4) {
468
+ throw new DecodeError("CONSTRAINT_VIOLATION", `Invalid filter_type: ${ft}`, r.offset);
469
+ }
470
+
471
+ let start_group: bigint | undefined;
472
+ let start_object: bigint | undefined;
473
+ let end_group: bigint | undefined;
474
+
475
+ if (ft >= 3) {
476
+ start_group = r.readVarInt();
477
+ start_object = r.readVarInt();
478
+ }
479
+ if (ft === 4) {
480
+ end_group = r.readVarInt();
481
+ }
482
+
483
+ const parameters = decodeParams(r);
484
+
485
+ const msg: Draft14Message & { type: "subscribe" } = {
486
+ type: "subscribe",
487
+ request_id,
488
+ track_namespace,
489
+ track_name,
490
+ subscriber_priority,
491
+ group_order,
492
+ forward,
493
+ filter_type,
494
+ parameters,
495
+ };
496
+
497
+ if (start_group !== undefined) {
498
+ return { ...msg, start_group, start_object, end_group } as Draft14Message;
499
+ }
500
+
501
+ return msg;
502
+ }
503
+
504
+ function decodeSubscribeOkPayload(r: BufferReader): Draft14Message {
505
+ const request_id = r.readVarInt();
506
+ const track_alias = r.readVarInt();
507
+ const expires = r.readVarInt();
508
+ const group_order = r.readVarInt();
509
+ const content_exists = r.readVarInt();
510
+
511
+ let largest_group: bigint | undefined;
512
+ let largest_object: bigint | undefined;
513
+
514
+ if (Number(content_exists) === 1) {
515
+ largest_group = r.readVarInt();
516
+ largest_object = r.readVarInt();
517
+ }
518
+
519
+ const parameters = decodeParams(r);
520
+
521
+ const msg: Draft14Message = {
522
+ type: "subscribe_ok",
523
+ request_id,
524
+ track_alias,
525
+ expires,
526
+ group_order,
527
+ content_exists,
528
+ parameters,
529
+ ...(largest_group !== undefined ? { largest_group, largest_object } : {}),
530
+ } as Draft14Message;
531
+
532
+ return msg;
533
+ }
534
+
535
+ function decodeSubscribeUpdatePayload(r: BufferReader): Draft14Message {
536
+ const request_id = r.readVarInt();
537
+ const start_group = r.readVarInt();
538
+ const start_object = r.readVarInt();
539
+ const end_group = r.readVarInt();
540
+ const subscriber_priority = BigInt(r.readUint8());
541
+ const forward = r.readVarInt();
542
+ const parameters = decodeParams(r);
543
+ return {
544
+ type: "subscribe_update",
545
+ request_id,
546
+ start_group,
547
+ start_object,
548
+ end_group,
549
+ subscriber_priority,
550
+ forward,
551
+ parameters,
552
+ };
553
+ }
554
+
555
+ function decodeSubscribeErrorPayload(r: BufferReader): Draft14Message {
556
+ const request_id = r.readVarInt();
557
+ const error_code = r.readVarInt();
558
+ const reason_phrase = r.readString();
559
+ return { type: "subscribe_error", request_id, error_code, reason_phrase };
560
+ }
561
+
562
+ function decodeUnsubscribePayload(r: BufferReader): Draft14Message {
563
+ const request_id = r.readVarInt();
564
+ return { type: "unsubscribe", request_id };
565
+ }
566
+
567
+ function decodePublishPayload(r: BufferReader): Draft14Message {
568
+ const request_id = r.readVarInt();
569
+ const track_namespace = r.readTuple();
570
+ const track_name = r.readString();
571
+ const forward = r.readVarInt();
572
+ const parameters = decodeParams(r);
573
+ return { type: "publish", request_id, track_namespace, track_name, forward, parameters };
574
+ }
575
+
576
+ function decodePublishOkPayload(r: BufferReader): Draft14Message {
577
+ const request_id = r.readVarInt();
578
+ const track_alias = r.readVarInt();
579
+ const forward = r.readVarInt();
580
+ const parameters = decodeParams(r);
581
+ return { type: "publish_ok", request_id, track_alias, forward, parameters };
582
+ }
583
+
584
+ function decodePublishErrorPayload(r: BufferReader): Draft14Message {
585
+ const request_id = r.readVarInt();
586
+ const error_code = r.readVarInt();
587
+ const reason_phrase = r.readString();
588
+ return { type: "publish_error", request_id, error_code, reason_phrase };
589
+ }
590
+
591
+ function decodePublishDonePayload(r: BufferReader): Draft14Message {
592
+ const request_id = r.readVarInt();
593
+ const status_code = r.readVarInt();
594
+ const reason_phrase = r.readString();
595
+ return { type: "publish_done", request_id, status_code, reason_phrase };
596
+ }
597
+
598
+ function decodePublishNamespacePayload(r: BufferReader): Draft14Message {
599
+ const request_id = r.readVarInt();
600
+ const track_namespace = r.readTuple();
601
+ const parameters = decodeParams(r);
602
+ return { type: "publish_namespace", request_id, track_namespace, parameters };
603
+ }
604
+
605
+ function decodePublishNamespaceOkPayload(r: BufferReader): Draft14Message {
606
+ const request_id = r.readVarInt();
607
+ const parameters = decodeParams(r);
608
+ return { type: "publish_namespace_ok", request_id, parameters };
609
+ }
610
+
611
+ function decodePublishNamespaceErrorPayload(r: BufferReader): Draft14Message {
612
+ const request_id = r.readVarInt();
613
+ const error_code = r.readVarInt();
614
+ const reason_phrase = r.readString();
615
+ return { type: "publish_namespace_error", request_id, error_code, reason_phrase };
616
+ }
617
+
618
+ function decodePublishNamespaceDonePayload(r: BufferReader): Draft14Message {
619
+ const request_id = r.readVarInt();
620
+ const status_code = r.readVarInt();
621
+ const reason_phrase = r.readString();
622
+ return { type: "publish_namespace_done", request_id, status_code, reason_phrase };
623
+ }
624
+
625
+ function decodePublishNamespaceCancelPayload(r: BufferReader): Draft14Message {
626
+ const request_id = r.readVarInt();
627
+ return { type: "publish_namespace_cancel", request_id };
628
+ }
629
+
630
+ function decodeSubscribeNamespacePayload(r: BufferReader): Draft14Message {
631
+ const request_id = r.readVarInt();
632
+ const namespace_prefix = r.readTuple();
633
+ const parameters = decodeParams(r);
634
+ return { type: "subscribe_namespace", request_id, namespace_prefix, parameters };
635
+ }
636
+
637
+ function decodeSubscribeNamespaceOkPayload(r: BufferReader): Draft14Message {
638
+ const request_id = r.readVarInt();
639
+ const parameters = decodeParams(r);
640
+ return { type: "subscribe_namespace_ok", request_id, parameters };
641
+ }
642
+
643
+ function decodeSubscribeNamespaceErrorPayload(r: BufferReader): Draft14Message {
644
+ const request_id = r.readVarInt();
645
+ const error_code = r.readVarInt();
646
+ const reason_phrase = r.readString();
647
+ return { type: "subscribe_namespace_error", request_id, error_code, reason_phrase };
648
+ }
649
+
650
+ function decodeUnsubscribeNamespacePayload(r: BufferReader): Draft14Message {
651
+ const request_id = r.readVarInt();
652
+ return { type: "unsubscribe_namespace", request_id };
653
+ }
654
+
655
+ function decodeFetchPayload(r: BufferReader): Draft14Message {
656
+ const request_id = r.readVarInt();
657
+ const track_namespace = r.readTuple();
658
+ const track_name = r.readString();
659
+ const start_group = r.readVarInt();
660
+ const start_object = r.readVarInt();
661
+ const end_group = r.readVarInt();
662
+ const parameters = decodeParams(r);
663
+ return {
664
+ type: "fetch",
665
+ request_id,
666
+ track_namespace,
667
+ track_name,
668
+ start_group,
669
+ start_object,
670
+ end_group,
671
+ parameters,
672
+ };
673
+ }
674
+
675
+ function decodeFetchOkPayload(r: BufferReader): Draft14Message {
676
+ const request_id = r.readVarInt();
677
+ const track_alias = r.readVarInt();
678
+ const end_of_track = r.readVarInt();
679
+ const parameters = decodeParams(r);
680
+ return { type: "fetch_ok", request_id, track_alias, end_of_track, parameters };
681
+ }
682
+
683
+ function decodeFetchErrorPayload(r: BufferReader): Draft14Message {
684
+ const request_id = r.readVarInt();
685
+ const error_code = r.readVarInt();
686
+ const reason_phrase = r.readString();
687
+ return { type: "fetch_error", request_id, error_code, reason_phrase };
688
+ }
689
+
690
+ function decodeFetchCancelPayload(r: BufferReader): Draft14Message {
691
+ const request_id = r.readVarInt();
692
+ return { type: "fetch_cancel", request_id };
693
+ }
694
+
695
+ function decodeTrackStatusPayload(r: BufferReader): Draft14Message {
696
+ const request_id = r.readVarInt();
697
+ const track_namespace = r.readTuple();
698
+ const track_name = r.readString();
699
+ const parameters = decodeParams(r);
700
+ return { type: "track_status", request_id, track_namespace, track_name, parameters };
701
+ }
702
+
703
+ function decodeTrackStatusOkPayload(r: BufferReader): Draft14Message {
704
+ const request_id = r.readVarInt();
705
+ const status_code = r.readVarInt();
706
+ const sc = Number(status_code);
707
+
708
+ let largest_group: bigint | undefined;
709
+ let largest_object: bigint | undefined;
710
+
711
+ if (sc === 0 || sc === 3) {
712
+ largest_group = r.readVarInt();
713
+ largest_object = r.readVarInt();
714
+ }
715
+
716
+ const parameters = decodeParams(r);
717
+
718
+ return {
719
+ type: "track_status_ok",
720
+ request_id,
721
+ status_code,
722
+ parameters,
723
+ ...(largest_group !== undefined ? { largest_group, largest_object } : {}),
724
+ } as Draft14Message;
725
+ }
726
+
727
+ function decodeTrackStatusErrorPayload(r: BufferReader): Draft14Message {
728
+ const request_id = r.readVarInt();
729
+ const error_code = r.readVarInt();
730
+ const reason_phrase = r.readString();
731
+ return { type: "track_status_error", request_id, error_code, reason_phrase };
732
+ }
733
+
734
+ function decodeGoAwayPayload(r: BufferReader): Draft14Message {
735
+ const new_session_uri = r.readString();
736
+ return { type: "goaway", new_session_uri };
737
+ }
738
+
739
+ function decodeMaxRequestIdPayload(r: BufferReader): Draft14Message {
740
+ const request_id = r.readVarInt();
741
+ return { type: "max_request_id", request_id };
742
+ }
743
+
744
+ function decodeRequestsBlockedPayload(r: BufferReader): Draft14Message {
745
+ const request_id = r.readVarInt();
746
+ return { type: "requests_blocked", request_id };
747
+ }
748
+
749
+ // ─── Payload dispatch tables ───────────────────────────────────────────────────
750
+
751
+ const payloadDecoders: ReadonlyMap<bigint, (r: BufferReader) => Draft14Message> = new Map([
752
+ [MSG_CLIENT_SETUP, decodeClientSetupPayload],
753
+ [MSG_SERVER_SETUP, decodeServerSetupPayload],
754
+ [MSG_SUBSCRIBE, decodeSubscribePayload],
755
+ [MSG_SUBSCRIBE_OK, decodeSubscribeOkPayload],
756
+ [MSG_SUBSCRIBE_UPDATE, decodeSubscribeUpdatePayload],
757
+ [MSG_SUBSCRIBE_ERROR, decodeSubscribeErrorPayload],
758
+ [MSG_UNSUBSCRIBE, decodeUnsubscribePayload],
759
+ [MSG_PUBLISH, decodePublishPayload],
760
+ [MSG_PUBLISH_OK, decodePublishOkPayload],
761
+ [MSG_PUBLISH_ERROR, decodePublishErrorPayload],
762
+ [MSG_PUBLISH_DONE, decodePublishDonePayload],
763
+ [MSG_PUBLISH_NAMESPACE, decodePublishNamespacePayload],
764
+ [MSG_PUBLISH_NAMESPACE_OK, decodePublishNamespaceOkPayload],
765
+ [MSG_PUBLISH_NAMESPACE_ERROR, decodePublishNamespaceErrorPayload],
766
+ [MSG_PUBLISH_NAMESPACE_DONE, decodePublishNamespaceDonePayload],
767
+ [MSG_PUBLISH_NAMESPACE_CANCEL, decodePublishNamespaceCancelPayload],
768
+ [MSG_SUBSCRIBE_NAMESPACE, decodeSubscribeNamespacePayload],
769
+ [MSG_SUBSCRIBE_NAMESPACE_OK, decodeSubscribeNamespaceOkPayload],
770
+ [MSG_SUBSCRIBE_NAMESPACE_ERROR, decodeSubscribeNamespaceErrorPayload],
771
+ [MSG_UNSUBSCRIBE_NAMESPACE, decodeUnsubscribeNamespacePayload],
772
+ [MSG_FETCH, decodeFetchPayload],
773
+ [MSG_FETCH_OK, decodeFetchOkPayload],
774
+ [MSG_FETCH_ERROR, decodeFetchErrorPayload],
775
+ [MSG_FETCH_CANCEL, decodeFetchCancelPayload],
776
+ [MSG_TRACK_STATUS, decodeTrackStatusPayload],
777
+ [MSG_TRACK_STATUS_OK, decodeTrackStatusOkPayload],
778
+ [MSG_TRACK_STATUS_ERROR, decodeTrackStatusErrorPayload],
779
+ [MSG_GOAWAY, decodeGoAwayPayload],
780
+ [MSG_MAX_REQUEST_ID, decodeMaxRequestIdPayload],
781
+ [MSG_REQUESTS_BLOCKED, decodeRequestsBlockedPayload],
782
+ ]);
783
+
784
+ // ─── Public API ────────────────────────────────────────────────────────────────
785
+
786
+ /**
787
+ * Encode a draft-14 control message with type(varint) + length(uint16 BE) + payload.
788
+ */
789
+ export function encodeMessage(message: Draft14Message): Uint8Array {
790
+ const typeId = MESSAGE_ID_MAP.get(message.type);
791
+ if (typeId === undefined) {
792
+ throw new Error(`Unknown message type: ${message.type}`);
793
+ }
794
+
795
+ // Encode payload into a separate buffer
796
+ const payloadWriter = new BufferWriter();
797
+ encodePayload(message, payloadWriter);
798
+ const payload = payloadWriter.finish();
799
+
800
+ if (payload.byteLength > 0xffff) {
801
+ throw new Error(`Payload too large for 16-bit length: ${payload.byteLength}`);
802
+ }
803
+
804
+ // Write framed message: type(varint) + length(uint16 BE) + payload
805
+ const writer = new BufferWriter();
806
+ writer.writeVarInt(typeId);
807
+ writer.writeUint8((payload.byteLength >> 8) & 0xff);
808
+ writer.writeUint8(payload.byteLength & 0xff);
809
+ writer.writeBytes(payload);
810
+
811
+ return writer.finish();
812
+ }
813
+
814
+ function encodePayload(msg: Draft14Message, w: BufferWriter): void {
815
+ switch (msg.type) {
816
+ case "client_setup":
817
+ return encodeClientSetupPayload(msg, w);
818
+ case "server_setup":
819
+ return encodeServerSetupPayload(msg, w);
820
+ case "subscribe":
821
+ return encodeSubscribePayload(msg, w);
822
+ case "subscribe_ok":
823
+ return encodeSubscribeOkPayload(msg, w);
824
+ case "subscribe_update":
825
+ return encodeSubscribeUpdatePayload(msg, w);
826
+ case "subscribe_error":
827
+ return encodeSubscribeErrorPayload(msg, w);
828
+ case "unsubscribe":
829
+ return encodeUnsubscribePayload(msg, w);
830
+ case "publish":
831
+ return encodePublishPayload(msg, w);
832
+ case "publish_ok":
833
+ return encodePublishOkPayload(msg, w);
834
+ case "publish_error":
835
+ return encodePublishErrorPayload(msg, w);
836
+ case "publish_done":
837
+ return encodePublishDonePayload(msg, w);
838
+ case "publish_namespace":
839
+ return encodePublishNamespacePayload(msg, w);
840
+ case "publish_namespace_ok":
841
+ return encodePublishNamespaceOkPayload(msg, w);
842
+ case "publish_namespace_error":
843
+ return encodePublishNamespaceErrorPayload(msg, w);
844
+ case "publish_namespace_done":
845
+ return encodePublishNamespaceDonePayload(msg, w);
846
+ case "publish_namespace_cancel":
847
+ return encodePublishNamespaceCancelPayload(msg, w);
848
+ case "subscribe_namespace":
849
+ return encodeSubscribeNamespacePayload(msg, w);
850
+ case "subscribe_namespace_ok":
851
+ return encodeSubscribeNamespaceOkPayload(msg, w);
852
+ case "subscribe_namespace_error":
853
+ return encodeSubscribeNamespaceErrorPayload(msg, w);
854
+ case "unsubscribe_namespace":
855
+ return encodeUnsubscribeNamespacePayload(msg, w);
856
+ case "fetch":
857
+ return encodeFetchPayload(msg, w);
858
+ case "fetch_ok":
859
+ return encodeFetchOkPayload(msg, w);
860
+ case "fetch_error":
861
+ return encodeFetchErrorPayload(msg, w);
862
+ case "fetch_cancel":
863
+ return encodeFetchCancelPayload(msg, w);
864
+ case "track_status":
865
+ return encodeTrackStatusPayload(msg, w);
866
+ case "track_status_ok":
867
+ return encodeTrackStatusOkPayload(msg, w);
868
+ case "track_status_error":
869
+ return encodeTrackStatusErrorPayload(msg, w);
870
+ case "goaway":
871
+ return encodeGoAwayPayload(msg, w);
872
+ case "max_request_id":
873
+ return encodeMaxRequestIdPayload(msg, w);
874
+ case "requests_blocked":
875
+ return encodeRequestsBlockedPayload(msg, w);
876
+ default: {
877
+ const _exhaustive: never = msg;
878
+ throw new Error(`Unhandled message type: ${(_exhaustive as Draft14Message).type}`);
879
+ }
880
+ }
881
+ }
882
+
883
+ /**
884
+ * Decode a draft-14 control message from bytes (type + uint16 length + payload).
885
+ */
886
+ export function decodeMessage(bytes: Uint8Array): DecodeResult<Draft14Message> {
887
+ try {
888
+ const reader = new BufferReader(bytes);
889
+ const typeId = reader.readVarInt();
890
+
891
+ // Read 16-bit big-endian payload length
892
+ const lenHi = reader.readUint8();
893
+ const lenLo = reader.readUint8();
894
+ const payloadLength = (lenHi << 8) | lenLo;
895
+
896
+ // Read exactly payloadLength bytes
897
+ const payloadBytes = reader.readBytes(payloadLength);
898
+ const payloadReader = new BufferReader(payloadBytes);
899
+
900
+ const decoder = payloadDecoders.get(typeId);
901
+ if (!decoder) {
902
+ return {
903
+ ok: false,
904
+ error: new DecodeError(
905
+ "UNKNOWN_MESSAGE_TYPE",
906
+ `Unknown message type ID: 0x${typeId.toString(16)}`,
907
+ 0,
908
+ ),
909
+ };
910
+ }
911
+
912
+ const message = decoder(payloadReader);
913
+ return { ok: true, value: message, bytesRead: reader.offset };
914
+ } catch (e) {
915
+ if (e instanceof DecodeError) {
916
+ return { ok: false, error: e };
917
+ }
918
+ throw e;
919
+ }
920
+ }
921
+
922
+ // ─── Data Stream Re-exports ───────────────────────────────────────────────────
923
+
924
+ import {
925
+ encodeSubgroupStream,
926
+ decodeSubgroupStream,
927
+ encodeDatagram,
928
+ decodeDatagram,
929
+ encodeFetchStream,
930
+ decodeFetchStream,
931
+ decodeDataStream,
932
+ createSubgroupStreamDecoder,
933
+ createFetchStreamDecoder,
934
+ createDataStreamDecoder,
935
+ } from "./data-streams.js";
936
+
937
+ export {
938
+ encodeSubgroupStream,
939
+ decodeSubgroupStream,
940
+ encodeDatagram,
941
+ decodeDatagram,
942
+ encodeFetchStream,
943
+ decodeFetchStream,
944
+ decodeDataStream,
945
+ createSubgroupStreamDecoder,
946
+ createFetchStreamDecoder,
947
+ createDataStreamDecoder,
948
+ };
949
+
950
+ // ─── Stream Decoders ───────────────────────────────────────────────────────────
951
+
952
+ /**
953
+ * Create a TransformStream that decodes a continuous byte stream (e.g. the
954
+ * WebTransport bidirectional control stream) into individual Draft14Message
955
+ * objects. Uses the varint(type) + uint16_BE(length) + payload framing.
956
+ */
957
+ export function createStreamDecoder(): TransformStream<Uint8Array, Draft14Message> {
958
+ let buffer = new Uint8Array(0);
959
+
960
+ return new TransformStream<Uint8Array, Draft14Message>({
961
+ transform(chunk, controller) {
962
+ // Accumulate incoming data
963
+ const newBuffer = new Uint8Array(buffer.length + chunk.length);
964
+ newBuffer.set(buffer, 0);
965
+ newBuffer.set(chunk, buffer.length);
966
+ buffer = newBuffer;
967
+
968
+ // Try to decode messages from the buffer
969
+ while (buffer.length > 0) {
970
+ const result = decodeMessage(buffer);
971
+ if (!result.ok) {
972
+ if (result.error.code === "UNEXPECTED_END") {
973
+ // Need more data -- wait for next chunk
974
+ break;
975
+ }
976
+ // Fatal decode error
977
+ controller.error(result.error);
978
+ return;
979
+ }
980
+ controller.enqueue(result.value);
981
+ // Advance the buffer past the consumed bytes
982
+ buffer = buffer.slice(result.bytesRead);
983
+ }
984
+ },
985
+
986
+ flush(controller) {
987
+ // If there is remaining data in the buffer, it is a truncated message
988
+ if (buffer.length > 0) {
989
+ controller.error(
990
+ new DecodeError("UNEXPECTED_END", "Stream ended with incomplete message data", 0),
991
+ );
992
+ }
993
+ },
994
+ });
995
+ }
996
+
997
+ // ─── Codec Factory ─────────────────────────────────────────────────────────────
998
+
999
+ export interface Draft14Codec extends BaseCodec<Draft14Message> {
1000
+ readonly draft: "draft-ietf-moq-transport-14";
1001
+ encodeSubgroupStream(stream: SubgroupStream): Uint8Array;
1002
+ encodeDatagram(dg: DatagramObject): Uint8Array;
1003
+ encodeFetchStream(stream: FetchStream): Uint8Array;
1004
+ decodeSubgroupStream(bytes: Uint8Array): DecodeResult<SubgroupStream>;
1005
+ decodeDatagram(bytes: Uint8Array): DecodeResult<DatagramObject>;
1006
+ decodeFetchStream(bytes: Uint8Array): DecodeResult<FetchStream>;
1007
+ decodeDataStream(
1008
+ streamType: "subgroup" | "datagram" | "fetch",
1009
+ bytes: Uint8Array,
1010
+ ): DecodeResult<Draft14DataStream>;
1011
+ createStreamDecoder(): TransformStream<Uint8Array, Draft14Message>;
1012
+ createSubgroupStreamDecoder(): TransformStream<Uint8Array, SubgroupStreamHeader | ObjectPayload>;
1013
+ createFetchStreamDecoder(): TransformStream<Uint8Array, FetchStreamHeader | ObjectPayload>;
1014
+ createDataStreamDecoder(): TransformStream<Uint8Array, DataStreamEvent>;
1015
+ }
1016
+
1017
+ export function createDraft14Codec(): Draft14Codec {
1018
+ return {
1019
+ draft: "draft-ietf-moq-transport-14",
1020
+ encodeMessage,
1021
+ decodeMessage,
1022
+ encodeSubgroupStream,
1023
+ encodeDatagram,
1024
+ encodeFetchStream,
1025
+ decodeSubgroupStream,
1026
+ decodeDatagram,
1027
+ decodeFetchStream,
1028
+ decodeDataStream,
1029
+ createStreamDecoder,
1030
+ createSubgroupStreamDecoder,
1031
+ createFetchStreamDecoder,
1032
+ createDataStreamDecoder,
1033
+ };
1034
+ }