@moqtap/codec 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (71) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +95 -0
  3. package/dist/chunk-23YG7F46.js +764 -0
  4. package/dist/chunk-2NARXGVA.cjs +194 -0
  5. package/dist/chunk-3BSZ55L3.cjs +307 -0
  6. package/dist/chunk-5WFXFLL4.cjs +1185 -0
  7. package/dist/chunk-DC4L6ZIT.js +307 -0
  8. package/dist/chunk-GDRGWFEK.cjs +498 -0
  9. package/dist/chunk-IQPDRQVC.js +1185 -0
  10. package/dist/chunk-QYG6KGOV.cjs +101 -0
  11. package/dist/chunk-UOBWHJA5.js +101 -0
  12. package/dist/chunk-WNTXF3DE.cjs +764 -0
  13. package/dist/chunk-YBSEOSSP.js +194 -0
  14. package/dist/chunk-YPXLV5YK.js +498 -0
  15. package/dist/codec-CTvFtQQI.d.cts +86 -0
  16. package/dist/codec-qPzfmLNu.d.ts +86 -0
  17. package/dist/draft14-session.cjs +6 -0
  18. package/dist/draft14-session.d.cts +8 -0
  19. package/dist/draft14-session.d.ts +8 -0
  20. package/dist/draft14-session.js +6 -0
  21. package/dist/draft14.cjs +121 -0
  22. package/dist/draft14.d.cts +96 -0
  23. package/dist/draft14.d.ts +96 -0
  24. package/dist/draft14.js +121 -0
  25. package/dist/draft7-session.cjs +7 -0
  26. package/dist/draft7-session.d.cts +7 -0
  27. package/dist/draft7-session.d.ts +7 -0
  28. package/dist/draft7-session.js +7 -0
  29. package/dist/draft7.cjs +60 -0
  30. package/dist/draft7.d.cts +72 -0
  31. package/dist/draft7.d.ts +72 -0
  32. package/dist/draft7.js +60 -0
  33. package/dist/index.cjs +40 -0
  34. package/dist/index.d.cts +40 -0
  35. package/dist/index.d.ts +40 -0
  36. package/dist/index.js +40 -0
  37. package/dist/session-types-B9NIf7_F.d.ts +101 -0
  38. package/dist/session-types-CCo-oA-d.d.cts +101 -0
  39. package/dist/session.cjs +27 -0
  40. package/dist/session.d.cts +24 -0
  41. package/dist/session.d.ts +24 -0
  42. package/dist/session.js +27 -0
  43. package/dist/types-CIk5W10V.d.cts +249 -0
  44. package/dist/types-CIk5W10V.d.ts +249 -0
  45. package/dist/types-ClXELFGN.d.cts +241 -0
  46. package/dist/types-ClXELFGN.d.ts +241 -0
  47. package/package.json +84 -0
  48. package/src/core/buffer-reader.ts +107 -0
  49. package/src/core/buffer-writer.ts +91 -0
  50. package/src/core/errors.ts +1 -0
  51. package/src/core/session-types.ts +103 -0
  52. package/src/core/types.ts +363 -0
  53. package/src/drafts/draft07/announce-fsm.ts +2 -0
  54. package/src/drafts/draft07/codec.ts +874 -0
  55. package/src/drafts/draft07/index.ts +70 -0
  56. package/src/drafts/draft07/messages.ts +44 -0
  57. package/src/drafts/draft07/parameters.ts +12 -0
  58. package/src/drafts/draft07/rules.ts +75 -0
  59. package/src/drafts/draft07/session-fsm.ts +353 -0
  60. package/src/drafts/draft07/session.ts +21 -0
  61. package/src/drafts/draft07/subscription-fsm.ts +3 -0
  62. package/src/drafts/draft07/varint.ts +23 -0
  63. package/src/drafts/draft14/codec.ts +1330 -0
  64. package/src/drafts/draft14/index.ts +132 -0
  65. package/src/drafts/draft14/messages.ts +76 -0
  66. package/src/drafts/draft14/rules.ts +70 -0
  67. package/src/drafts/draft14/session-fsm.ts +480 -0
  68. package/src/drafts/draft14/session.ts +26 -0
  69. package/src/drafts/draft14/types.ts +365 -0
  70. package/src/index.ts +85 -0
  71. package/src/session.ts +58 -0
@@ -0,0 +1,874 @@
1
+ import { BufferReader } from '../../core/buffer-reader.js';
2
+ import { BufferWriter } from '../../core/buffer-writer.js';
3
+ import { DecodeError } from '../../core/types.js';
4
+ import type {
5
+ Codec,
6
+ DecodeResult,
7
+ FilterType,
8
+ GroupOrderValue,
9
+ MoqtMessage,
10
+ MoqtMessageType,
11
+ ClientSetup,
12
+ ServerSetup,
13
+ Subscribe,
14
+ SubscribeOk,
15
+ SubscribeError,
16
+ SubscribeDone,
17
+ SubscribeUpdate,
18
+ Unsubscribe,
19
+ Announce,
20
+ AnnounceOk,
21
+ AnnounceError,
22
+ AnnounceCancel,
23
+ Unannounce,
24
+ TrackStatusRequest,
25
+ TrackStatus,
26
+ ObjectStream,
27
+ ObjectDatagram,
28
+ StreamHeaderTrack,
29
+ StreamHeaderGroup,
30
+ StreamHeaderSubgroup,
31
+ GoAway,
32
+ SubscribeAnnounces,
33
+ SubscribeAnnouncesOk,
34
+ SubscribeAnnouncesError,
35
+ UnsubscribeAnnounces,
36
+ MaxSubscribeId,
37
+ Fetch,
38
+ FetchOk,
39
+ FetchError,
40
+ FetchCancel,
41
+ } from '../../core/types.js';
42
+ import { MESSAGE_TYPE_IDS } from './messages.js';
43
+ import { encodeVarInt, decodeVarInt } from './varint.js';
44
+
45
+ // --- FilterType mapping ---
46
+
47
+ const FILTER_TYPE_TO_WIRE: Record<FilterType, bigint> = {
48
+ latest_group: 1n,
49
+ latest_object: 2n,
50
+ absolute_start: 3n,
51
+ absolute_range: 4n,
52
+ };
53
+
54
+ const WIRE_TO_FILTER_TYPE: Map<bigint, FilterType> = new Map([
55
+ [1n, 'latest_group'],
56
+ [2n, 'latest_object'],
57
+ [3n, 'absolute_start'],
58
+ [4n, 'absolute_range'],
59
+ ]);
60
+
61
+ // --- GroupOrderValue mapping ---
62
+
63
+ const GROUP_ORDER_TO_WIRE: Record<GroupOrderValue, number> = {
64
+ original: 0,
65
+ ascending: 1,
66
+ descending: 2,
67
+ };
68
+
69
+ const WIRE_TO_GROUP_ORDER: Map<number, GroupOrderValue> = new Map([
70
+ [0, 'original'],
71
+ [1, 'ascending'],
72
+ [2, 'descending'],
73
+ ]);
74
+
75
+ // --- Encode helpers ---
76
+
77
+ function writeGroupOrder(writer: BufferWriter, value: GroupOrderValue): void {
78
+ const wire = GROUP_ORDER_TO_WIRE[value];
79
+ writer.writeUint8(wire);
80
+ }
81
+
82
+ function readGroupOrder(reader: BufferReader): GroupOrderValue {
83
+ const wire = reader.readUint8();
84
+ const value = WIRE_TO_GROUP_ORDER.get(wire);
85
+ if (value === undefined) {
86
+ throw new DecodeError('CONSTRAINT_VIOLATION', `Invalid group order value: ${wire}`, reader.offset - 1);
87
+ }
88
+ return value;
89
+ }
90
+
91
+ // Data stream type IDs that NEVER appear as control messages.
92
+ // Note: 0x04 (stream_header_subgroup) excluded — shares ID with subscribe_ok.
93
+ // Callers must use dataStreamDecoders directly for stream_header_subgroup.
94
+ const DATA_STREAM_TYPE_IDS: ReadonlySet<bigint> = new Set([
95
+ MESSAGE_TYPE_IDS.object_stream,
96
+ MESSAGE_TYPE_IDS.object_datagram,
97
+ MESSAGE_TYPE_IDS.stream_header_track,
98
+ MESSAGE_TYPE_IDS.stream_header_group,
99
+ ]);
100
+
101
+ // --- Control message type (excludes data stream types) ---
102
+ type ControlMessageType = Exclude<
103
+ MoqtMessageType,
104
+ 'object_stream' | 'object_datagram' | 'stream_header_track' | 'stream_header_group' | 'stream_header_subgroup'
105
+ >;
106
+
107
+ // --- Encode functions for each message type (payload only, no type ID) ---
108
+
109
+ function encodeClientSetup(msg: ClientSetup, writer: BufferWriter): void {
110
+ writer.writeVarInt(msg.supportedVersions.length);
111
+ for (const version of msg.supportedVersions) {
112
+ writer.writeVarInt(version);
113
+ }
114
+ writer.writeParameters(msg.parameters);
115
+ }
116
+
117
+ function encodeServerSetup(msg: ServerSetup, writer: BufferWriter): void {
118
+ writer.writeVarInt(msg.selectedVersion);
119
+ writer.writeParameters(msg.parameters);
120
+ }
121
+
122
+ function encodeSubscribe(msg: Subscribe, writer: BufferWriter): void {
123
+ writer.writeVarInt(msg.subscribeId);
124
+ writer.writeVarInt(msg.trackAlias);
125
+ writer.writeTuple(msg.trackNamespace);
126
+ writer.writeString(msg.trackName);
127
+ writer.writeUint8(msg.subscriberPriority);
128
+ writeGroupOrder(writer, msg.groupOrder);
129
+ writer.writeVarInt(FILTER_TYPE_TO_WIRE[msg.filterType]);
130
+ if (msg.filterType === 'absolute_start' || msg.filterType === 'absolute_range') {
131
+ writer.writeVarInt(msg.startGroup!);
132
+ writer.writeVarInt(msg.startObject!);
133
+ }
134
+ if (msg.filterType === 'absolute_range') {
135
+ writer.writeVarInt(msg.endGroup!);
136
+ writer.writeVarInt(msg.endObject!);
137
+ }
138
+ writer.writeParameters(msg.parameters);
139
+ }
140
+
141
+ function encodeSubscribeOk(msg: SubscribeOk, writer: BufferWriter): void {
142
+ writer.writeVarInt(msg.subscribeId);
143
+ writer.writeVarInt(msg.expires);
144
+ writeGroupOrder(writer, msg.groupOrder);
145
+ writer.writeUint8(msg.contentExists ? 1 : 0);
146
+ if (msg.contentExists) {
147
+ writer.writeVarInt(msg.largestGroupId!);
148
+ writer.writeVarInt(msg.largestObjectId!);
149
+ }
150
+ writer.writeParameters(msg.parameters);
151
+ }
152
+
153
+ function encodeSubscribeError(msg: SubscribeError, writer: BufferWriter): void {
154
+ writer.writeVarInt(msg.subscribeId);
155
+ writer.writeVarInt(msg.errorCode);
156
+ writer.writeString(msg.reasonPhrase);
157
+ writer.writeVarInt(msg.trackAlias);
158
+ }
159
+
160
+ function encodeSubscribeDone(msg: SubscribeDone, writer: BufferWriter): void {
161
+ writer.writeVarInt(msg.subscribeId);
162
+ writer.writeVarInt(msg.statusCode);
163
+ writer.writeString(msg.reasonPhrase);
164
+ writer.writeUint8(msg.contentExists ? 1 : 0);
165
+ if (msg.contentExists) {
166
+ writer.writeVarInt(msg.finalGroupId!);
167
+ writer.writeVarInt(msg.finalObjectId!);
168
+ }
169
+ }
170
+
171
+ function encodeSubscribeUpdate(msg: SubscribeUpdate, writer: BufferWriter): void {
172
+ writer.writeVarInt(msg.subscribeId);
173
+ writer.writeVarInt(msg.startGroup);
174
+ writer.writeVarInt(msg.startObject);
175
+ writer.writeVarInt(msg.endGroup);
176
+ writer.writeVarInt(msg.endObject);
177
+ writer.writeUint8(msg.subscriberPriority);
178
+ writer.writeParameters(msg.parameters);
179
+ }
180
+
181
+ function encodeUnsubscribe(msg: Unsubscribe, writer: BufferWriter): void {
182
+ writer.writeVarInt(msg.subscribeId);
183
+ }
184
+
185
+ function encodeAnnounce(msg: Announce, writer: BufferWriter): void {
186
+ writer.writeTuple(msg.trackNamespace);
187
+ writer.writeParameters(msg.parameters);
188
+ }
189
+
190
+ function encodeAnnounceOk(msg: AnnounceOk, writer: BufferWriter): void {
191
+ writer.writeTuple(msg.trackNamespace);
192
+ }
193
+
194
+ function encodeAnnounceError(msg: AnnounceError, writer: BufferWriter): void {
195
+ writer.writeTuple(msg.trackNamespace);
196
+ writer.writeVarInt(msg.errorCode);
197
+ writer.writeString(msg.reasonPhrase);
198
+ }
199
+
200
+ function encodeAnnounceCancel(msg: AnnounceCancel, writer: BufferWriter): void {
201
+ writer.writeTuple(msg.trackNamespace);
202
+ writer.writeVarInt(msg.errorCode);
203
+ writer.writeString(msg.reasonPhrase);
204
+ }
205
+
206
+ function encodeUnannounce(msg: Unannounce, writer: BufferWriter): void {
207
+ writer.writeTuple(msg.trackNamespace);
208
+ }
209
+
210
+ function encodeTrackStatusRequest(msg: TrackStatusRequest, writer: BufferWriter): void {
211
+ writer.writeTuple(msg.trackNamespace);
212
+ writer.writeString(msg.trackName);
213
+ }
214
+
215
+ function encodeTrackStatus(msg: TrackStatus, writer: BufferWriter): void {
216
+ writer.writeTuple(msg.trackNamespace);
217
+ writer.writeString(msg.trackName);
218
+ writer.writeVarInt(msg.statusCode);
219
+ writer.writeVarInt(msg.lastGroupId);
220
+ writer.writeVarInt(msg.lastObjectId);
221
+ }
222
+
223
+ function encodeGoAway(msg: GoAway, writer: BufferWriter): void {
224
+ writer.writeString(msg.newSessionUri);
225
+ }
226
+
227
+ function encodeSubscribeAnnounces(msg: SubscribeAnnounces, writer: BufferWriter): void {
228
+ writer.writeTuple(msg.trackNamespace);
229
+ writer.writeParameters(msg.parameters);
230
+ }
231
+
232
+ function encodeSubscribeAnnouncesOk(msg: SubscribeAnnouncesOk, writer: BufferWriter): void {
233
+ writer.writeTuple(msg.trackNamespace);
234
+ }
235
+
236
+ function encodeSubscribeAnnouncesError(msg: SubscribeAnnouncesError, writer: BufferWriter): void {
237
+ writer.writeTuple(msg.trackNamespace);
238
+ writer.writeVarInt(msg.errorCode);
239
+ writer.writeString(msg.reasonPhrase);
240
+ }
241
+
242
+ function encodeUnsubscribeAnnounces(msg: UnsubscribeAnnounces, writer: BufferWriter): void {
243
+ writer.writeTuple(msg.trackNamespace);
244
+ }
245
+
246
+ function encodeMaxSubscribeId(msg: MaxSubscribeId, writer: BufferWriter): void {
247
+ writer.writeVarInt(msg.subscribeId);
248
+ }
249
+
250
+ function encodeFetch(msg: Fetch, writer: BufferWriter): void {
251
+ writer.writeVarInt(msg.subscribeId);
252
+ writer.writeTuple(msg.trackNamespace);
253
+ writer.writeString(msg.trackName);
254
+ writer.writeUint8(msg.subscriberPriority);
255
+ writeGroupOrder(writer, msg.groupOrder);
256
+ writer.writeVarInt(msg.startGroup);
257
+ writer.writeVarInt(msg.startObject);
258
+ writer.writeVarInt(msg.endGroup);
259
+ writer.writeVarInt(msg.endObject);
260
+ writer.writeParameters(msg.parameters);
261
+ }
262
+
263
+ function encodeFetchOk(msg: FetchOk, writer: BufferWriter): void {
264
+ writer.writeVarInt(msg.subscribeId);
265
+ writeGroupOrder(writer, msg.groupOrder);
266
+ writer.writeUint8(msg.endOfTrack ? 1 : 0);
267
+ writer.writeVarInt(msg.largestGroupId);
268
+ writer.writeVarInt(msg.largestObjectId);
269
+ writer.writeParameters(msg.parameters);
270
+ }
271
+
272
+ function encodeFetchError(msg: FetchError, writer: BufferWriter): void {
273
+ writer.writeVarInt(msg.subscribeId);
274
+ writer.writeVarInt(msg.errorCode);
275
+ writer.writeString(msg.reasonPhrase);
276
+ }
277
+
278
+ function encodeFetchCancel(msg: FetchCancel, writer: BufferWriter): void {
279
+ writer.writeVarInt(msg.subscribeId);
280
+ }
281
+
282
+ // Data stream encoders (no type+length framing)
283
+ function encodeObjectStream(msg: ObjectStream, writer: BufferWriter): void {
284
+ writer.writeVarInt(MESSAGE_TYPE_IDS.object_stream);
285
+ writer.writeVarInt(msg.subscribeId);
286
+ writer.writeVarInt(msg.trackAlias);
287
+ writer.writeVarInt(msg.groupId);
288
+ writer.writeVarInt(msg.objectId);
289
+ writer.writeUint8(msg.publisherPriority);
290
+ if (msg.objectStatus !== undefined) {
291
+ writer.writeVarInt(msg.objectStatus);
292
+ } else {
293
+ writer.writeVarInt(0);
294
+ }
295
+ writer.writeBytes(msg.payload);
296
+ }
297
+
298
+ function encodeObjectDatagram(msg: ObjectDatagram, writer: BufferWriter): void {
299
+ writer.writeVarInt(MESSAGE_TYPE_IDS.object_datagram);
300
+ writer.writeVarInt(msg.subscribeId);
301
+ writer.writeVarInt(msg.trackAlias);
302
+ writer.writeVarInt(msg.groupId);
303
+ writer.writeVarInt(msg.objectId);
304
+ writer.writeUint8(msg.publisherPriority);
305
+ if (msg.objectStatus !== undefined) {
306
+ writer.writeVarInt(msg.objectStatus);
307
+ } else {
308
+ writer.writeVarInt(0);
309
+ }
310
+ writer.writeBytes(msg.payload);
311
+ }
312
+
313
+ function encodeStreamHeaderTrack(msg: StreamHeaderTrack, writer: BufferWriter): void {
314
+ writer.writeVarInt(MESSAGE_TYPE_IDS.stream_header_track);
315
+ writer.writeVarInt(msg.subscribeId);
316
+ writer.writeVarInt(msg.trackAlias);
317
+ writer.writeUint8(msg.publisherPriority);
318
+ }
319
+
320
+ function encodeStreamHeaderGroup(msg: StreamHeaderGroup, writer: BufferWriter): void {
321
+ writer.writeVarInt(MESSAGE_TYPE_IDS.stream_header_group);
322
+ writer.writeVarInt(msg.subscribeId);
323
+ writer.writeVarInt(msg.trackAlias);
324
+ writer.writeVarInt(msg.groupId);
325
+ writer.writeUint8(msg.publisherPriority);
326
+ }
327
+
328
+ function encodeStreamHeaderSubgroup(msg: StreamHeaderSubgroup, writer: BufferWriter): void {
329
+ writer.writeVarInt(MESSAGE_TYPE_IDS.stream_header_subgroup);
330
+ writer.writeVarInt(msg.subscribeId);
331
+ writer.writeVarInt(msg.trackAlias);
332
+ writer.writeVarInt(msg.groupId);
333
+ writer.writeVarInt(msg.subgroupId);
334
+ writer.writeUint8(msg.publisherPriority);
335
+ }
336
+
337
+ // --- Encode dispatch ---
338
+
339
+ // Control message encoders (payload-only, framing added by encodeMessageImpl)
340
+ const controlEncoders: Record<ControlMessageType, (msg: never, writer: BufferWriter) => void> = {
341
+ client_setup: encodeClientSetup as (msg: never, writer: BufferWriter) => void,
342
+ server_setup: encodeServerSetup as (msg: never, writer: BufferWriter) => void,
343
+ subscribe: encodeSubscribe as (msg: never, writer: BufferWriter) => void,
344
+ subscribe_ok: encodeSubscribeOk as (msg: never, writer: BufferWriter) => void,
345
+ subscribe_error: encodeSubscribeError as (msg: never, writer: BufferWriter) => void,
346
+ subscribe_done: encodeSubscribeDone as (msg: never, writer: BufferWriter) => void,
347
+ subscribe_update: encodeSubscribeUpdate as (msg: never, writer: BufferWriter) => void,
348
+ unsubscribe: encodeUnsubscribe as (msg: never, writer: BufferWriter) => void,
349
+ announce: encodeAnnounce as (msg: never, writer: BufferWriter) => void,
350
+ announce_ok: encodeAnnounceOk as (msg: never, writer: BufferWriter) => void,
351
+ announce_error: encodeAnnounceError as (msg: never, writer: BufferWriter) => void,
352
+ announce_cancel: encodeAnnounceCancel as (msg: never, writer: BufferWriter) => void,
353
+ unannounce: encodeUnannounce as (msg: never, writer: BufferWriter) => void,
354
+ track_status_request: encodeTrackStatusRequest as (msg: never, writer: BufferWriter) => void,
355
+ track_status: encodeTrackStatus as (msg: never, writer: BufferWriter) => void,
356
+ goaway: encodeGoAway as (msg: never, writer: BufferWriter) => void,
357
+ subscribe_announces: encodeSubscribeAnnounces as (msg: never, writer: BufferWriter) => void,
358
+ subscribe_announces_ok: encodeSubscribeAnnouncesOk as (msg: never, writer: BufferWriter) => void,
359
+ subscribe_announces_error: encodeSubscribeAnnouncesError as (msg: never, writer: BufferWriter) => void,
360
+ unsubscribe_announces: encodeUnsubscribeAnnounces as (msg: never, writer: BufferWriter) => void,
361
+ max_subscribe_id: encodeMaxSubscribeId as (msg: never, writer: BufferWriter) => void,
362
+ fetch: encodeFetch as (msg: never, writer: BufferWriter) => void,
363
+ fetch_ok: encodeFetchOk as (msg: never, writer: BufferWriter) => void,
364
+ fetch_error: encodeFetchError as (msg: never, writer: BufferWriter) => void,
365
+ fetch_cancel: encodeFetchCancel as (msg: never, writer: BufferWriter) => void,
366
+ };
367
+
368
+ // Data stream encoders (write type + fields directly, no length framing)
369
+ const dataStreamEncoders: Partial<Record<MoqtMessageType, (msg: never, writer: BufferWriter) => void>> = {
370
+ object_stream: encodeObjectStream as (msg: never, writer: BufferWriter) => void,
371
+ object_datagram: encodeObjectDatagram as (msg: never, writer: BufferWriter) => void,
372
+ stream_header_track: encodeStreamHeaderTrack as (msg: never, writer: BufferWriter) => void,
373
+ stream_header_group: encodeStreamHeaderGroup as (msg: never, writer: BufferWriter) => void,
374
+ stream_header_subgroup: encodeStreamHeaderSubgroup as (msg: never, writer: BufferWriter) => void,
375
+ };
376
+
377
+ // --- Decode functions for each message type ---
378
+
379
+ function decodeClientSetup(reader: BufferReader): ClientSetup {
380
+ const numVersions = reader.readVarInt();
381
+ if (numVersions === 0n) {
382
+ throw new DecodeError('CONSTRAINT_VIOLATION', 'supported_versions must not be empty', reader.offset);
383
+ }
384
+ const supportedVersions: bigint[] = [];
385
+ for (let i = 0n; i < numVersions; i++) {
386
+ supportedVersions.push(reader.readVarInt());
387
+ }
388
+ const parameters = reader.readParameters();
389
+ return { type: 'client_setup', supportedVersions, parameters };
390
+ }
391
+
392
+ function decodeServerSetup(reader: BufferReader): ServerSetup {
393
+ const selectedVersion = reader.readVarInt();
394
+ const parameters = reader.readParameters();
395
+ return { type: 'server_setup', selectedVersion, parameters };
396
+ }
397
+
398
+ function decodeSubscribe(reader: BufferReader): Subscribe {
399
+ const subscribeId = reader.readVarInt();
400
+ const trackAlias = reader.readVarInt();
401
+ const trackNamespace = reader.readTuple();
402
+ const trackName = reader.readString();
403
+ const subscriberPriority = reader.readUint8();
404
+ const groupOrder = readGroupOrder(reader);
405
+ const filterTypeWire = reader.readVarInt();
406
+ const filterType = WIRE_TO_FILTER_TYPE.get(filterTypeWire);
407
+ if (filterType === undefined) {
408
+ throw new DecodeError('CONSTRAINT_VIOLATION', `Invalid filter type: ${filterTypeWire}`, reader.offset);
409
+ }
410
+
411
+ const base = {
412
+ type: 'subscribe' as const,
413
+ subscribeId,
414
+ trackAlias,
415
+ trackNamespace,
416
+ trackName,
417
+ subscriberPriority,
418
+ groupOrder,
419
+ filterType,
420
+ parameters: undefined as unknown as Map<bigint, Uint8Array>,
421
+ };
422
+
423
+ if (filterType === 'absolute_start') {
424
+ const startGroup = reader.readVarInt();
425
+ const startObject = reader.readVarInt();
426
+ base.parameters = reader.readParameters();
427
+ return { ...base, startGroup, startObject };
428
+ }
429
+ if (filterType === 'absolute_range') {
430
+ const startGroup = reader.readVarInt();
431
+ const startObject = reader.readVarInt();
432
+ const endGroup = reader.readVarInt();
433
+ const endObject = reader.readVarInt();
434
+ base.parameters = reader.readParameters();
435
+ return { ...base, startGroup, startObject, endGroup, endObject };
436
+ }
437
+
438
+ base.parameters = reader.readParameters();
439
+ return base;
440
+ }
441
+
442
+ function decodeSubscribeOk(reader: BufferReader): SubscribeOk {
443
+ const subscribeId = reader.readVarInt();
444
+ const expires = reader.readVarInt();
445
+ const groupOrder = readGroupOrder(reader);
446
+ const contentExistsWire = reader.readUint8();
447
+ const contentExists = contentExistsWire !== 0;
448
+
449
+ if (contentExists) {
450
+ const largestGroupId = reader.readVarInt();
451
+ const largestObjectId = reader.readVarInt();
452
+ const parameters = reader.readParameters();
453
+ return { type: 'subscribe_ok' as const, subscribeId, expires, groupOrder, contentExists, largestGroupId, largestObjectId, parameters };
454
+ }
455
+
456
+ const parameters = reader.readParameters();
457
+ return { type: 'subscribe_ok' as const, subscribeId, expires, groupOrder, contentExists, parameters };
458
+ }
459
+
460
+ function decodeSubscribeError(reader: BufferReader): SubscribeError {
461
+ const subscribeId = reader.readVarInt();
462
+ const errorCode = reader.readVarInt();
463
+ const reasonPhrase = reader.readString();
464
+ const trackAlias = reader.readVarInt();
465
+ return { type: 'subscribe_error', subscribeId, errorCode, reasonPhrase, trackAlias };
466
+ }
467
+
468
+ function decodeSubscribeDone(reader: BufferReader): SubscribeDone {
469
+ const subscribeId = reader.readVarInt();
470
+ const statusCode = reader.readVarInt();
471
+ const reasonPhrase = reader.readString();
472
+ const contentExistsWire = reader.readUint8();
473
+ const contentExists = contentExistsWire !== 0;
474
+
475
+ if (contentExists) {
476
+ const finalGroupId = reader.readVarInt();
477
+ const finalObjectId = reader.readVarInt();
478
+ return { type: 'subscribe_done' as const, subscribeId, statusCode, reasonPhrase, contentExists, finalGroupId, finalObjectId };
479
+ }
480
+
481
+ return { type: 'subscribe_done' as const, subscribeId, statusCode, reasonPhrase, contentExists };
482
+ }
483
+
484
+ function decodeSubscribeUpdate(reader: BufferReader): SubscribeUpdate {
485
+ const subscribeId = reader.readVarInt();
486
+ const startGroup = reader.readVarInt();
487
+ const startObject = reader.readVarInt();
488
+ const endGroup = reader.readVarInt();
489
+ const endObject = reader.readVarInt();
490
+ const subscriberPriority = reader.readUint8();
491
+ const parameters = reader.readParameters();
492
+ return { type: 'subscribe_update', subscribeId, startGroup, startObject, endGroup, endObject, subscriberPriority, parameters };
493
+ }
494
+
495
+ function decodeUnsubscribe(reader: BufferReader): Unsubscribe {
496
+ const subscribeId = reader.readVarInt();
497
+ return { type: 'unsubscribe', subscribeId };
498
+ }
499
+
500
+ function decodeAnnounce(reader: BufferReader): Announce {
501
+ const trackNamespace = reader.readTuple();
502
+ const parameters = reader.readParameters();
503
+ return { type: 'announce', trackNamespace, parameters };
504
+ }
505
+
506
+ function decodeAnnounceOk(reader: BufferReader): AnnounceOk {
507
+ const trackNamespace = reader.readTuple();
508
+ return { type: 'announce_ok', trackNamespace };
509
+ }
510
+
511
+ function decodeAnnounceError(reader: BufferReader): AnnounceError {
512
+ const trackNamespace = reader.readTuple();
513
+ const errorCode = reader.readVarInt();
514
+ const reasonPhrase = reader.readString();
515
+ return { type: 'announce_error', trackNamespace, errorCode, reasonPhrase };
516
+ }
517
+
518
+ function decodeAnnounceCancel(reader: BufferReader): AnnounceCancel {
519
+ const trackNamespace = reader.readTuple();
520
+ const errorCode = reader.readVarInt();
521
+ const reasonPhrase = reader.readString();
522
+ return { type: 'announce_cancel', trackNamespace, errorCode, reasonPhrase };
523
+ }
524
+
525
+ function decodeUnannounce(reader: BufferReader): Unannounce {
526
+ const trackNamespace = reader.readTuple();
527
+ return { type: 'unannounce', trackNamespace };
528
+ }
529
+
530
+ function decodeTrackStatusRequest(reader: BufferReader): TrackStatusRequest {
531
+ const trackNamespace = reader.readTuple();
532
+ const trackName = reader.readString();
533
+ return { type: 'track_status_request', trackNamespace, trackName };
534
+ }
535
+
536
+ function decodeTrackStatus(reader: BufferReader): TrackStatus {
537
+ const trackNamespace = reader.readTuple();
538
+ const trackName = reader.readString();
539
+ const statusCode = reader.readVarInt();
540
+ const lastGroupId = reader.readVarInt();
541
+ const lastObjectId = reader.readVarInt();
542
+ return { type: 'track_status', trackNamespace, trackName, statusCode, lastGroupId, lastObjectId };
543
+ }
544
+
545
+ function decodeGoAway(reader: BufferReader): GoAway {
546
+ const newSessionUri = reader.readString();
547
+ return { type: 'goaway', newSessionUri };
548
+ }
549
+
550
+ function decodeSubscribeAnnounces(reader: BufferReader): SubscribeAnnounces {
551
+ const trackNamespace = reader.readTuple();
552
+ const parameters = reader.readParameters();
553
+ return { type: 'subscribe_announces', trackNamespace, parameters };
554
+ }
555
+
556
+ function decodeSubscribeAnnouncesOk(reader: BufferReader): SubscribeAnnouncesOk {
557
+ const trackNamespace = reader.readTuple();
558
+ return { type: 'subscribe_announces_ok', trackNamespace };
559
+ }
560
+
561
+ function decodeSubscribeAnnouncesError(reader: BufferReader): SubscribeAnnouncesError {
562
+ const trackNamespace = reader.readTuple();
563
+ const errorCode = reader.readVarInt();
564
+ const reasonPhrase = reader.readString();
565
+ return { type: 'subscribe_announces_error', trackNamespace, errorCode, reasonPhrase };
566
+ }
567
+
568
+ function decodeUnsubscribeAnnounces(reader: BufferReader): UnsubscribeAnnounces {
569
+ const trackNamespace = reader.readTuple();
570
+ return { type: 'unsubscribe_announces', trackNamespace };
571
+ }
572
+
573
+ function decodeMaxSubscribeId(reader: BufferReader): MaxSubscribeId {
574
+ const subscribeId = reader.readVarInt();
575
+ return { type: 'max_subscribe_id', subscribeId };
576
+ }
577
+
578
+ function decodeFetch(reader: BufferReader): Fetch {
579
+ const subscribeId = reader.readVarInt();
580
+ const trackNamespace = reader.readTuple();
581
+ const trackName = reader.readString();
582
+ const subscriberPriority = reader.readUint8();
583
+ const groupOrder = readGroupOrder(reader);
584
+ const startGroup = reader.readVarInt();
585
+ const startObject = reader.readVarInt();
586
+ const endGroup = reader.readVarInt();
587
+ const endObject = reader.readVarInt();
588
+ const parameters = reader.readParameters();
589
+ return { type: 'fetch' as const, subscribeId, trackNamespace, trackName, subscriberPriority, groupOrder, startGroup, startObject, endGroup, endObject, parameters };
590
+ }
591
+
592
+ function decodeFetchOk(reader: BufferReader): FetchOk {
593
+ const subscribeId = reader.readVarInt();
594
+ const groupOrder = readGroupOrder(reader);
595
+ const endOfTrackWire = reader.readUint8();
596
+ const endOfTrack = endOfTrackWire !== 0;
597
+ const largestGroupId = reader.readVarInt();
598
+ const largestObjectId = reader.readVarInt();
599
+ const parameters = reader.readParameters();
600
+ return { type: 'fetch_ok' as const, subscribeId, groupOrder, endOfTrack, largestGroupId, largestObjectId, parameters };
601
+ }
602
+
603
+ function decodeFetchError(reader: BufferReader): FetchError {
604
+ const subscribeId = reader.readVarInt();
605
+ const errorCode = reader.readVarInt();
606
+ const reasonPhrase = reader.readString();
607
+ return { type: 'fetch_error', subscribeId, errorCode, reasonPhrase };
608
+ }
609
+
610
+ function decodeFetchCancel(reader: BufferReader): FetchCancel {
611
+ const subscribeId = reader.readVarInt();
612
+ return { type: 'fetch_cancel', subscribeId };
613
+ }
614
+
615
+ function decodeObjectStream(reader: BufferReader): ObjectStream {
616
+ const subscribeId = reader.readVarInt();
617
+ const trackAlias = reader.readVarInt();
618
+ const groupId = reader.readVarInt();
619
+ const objectId = reader.readVarInt();
620
+ const publisherPriority = reader.readUint8();
621
+ const objectStatusRaw = Number(reader.readVarInt());
622
+ const payload = reader.readBytes(reader.remaining);
623
+ const base = { type: 'object_stream' as const, subscribeId, trackAlias, groupId, objectId, publisherPriority, payload };
624
+ if (objectStatusRaw !== 0) {
625
+ return { ...base, objectStatus: objectStatusRaw };
626
+ }
627
+ return base;
628
+ }
629
+
630
+ function decodeObjectDatagram(reader: BufferReader): ObjectDatagram {
631
+ const subscribeId = reader.readVarInt();
632
+ const trackAlias = reader.readVarInt();
633
+ const groupId = reader.readVarInt();
634
+ const objectId = reader.readVarInt();
635
+ const publisherPriority = reader.readUint8();
636
+ const objectStatusRaw = Number(reader.readVarInt());
637
+ const payload = reader.readBytes(reader.remaining);
638
+ const base = { type: 'object_datagram' as const, subscribeId, trackAlias, groupId, objectId, publisherPriority, payload };
639
+ if (objectStatusRaw !== 0) {
640
+ return { ...base, objectStatus: objectStatusRaw };
641
+ }
642
+ return base;
643
+ }
644
+
645
+ function decodeStreamHeaderTrack(reader: BufferReader): StreamHeaderTrack {
646
+ const subscribeId = reader.readVarInt();
647
+ const trackAlias = reader.readVarInt();
648
+ const publisherPriority = reader.readUint8();
649
+ return { type: 'stream_header_track', subscribeId, trackAlias, publisherPriority };
650
+ }
651
+
652
+ function decodeStreamHeaderGroup(reader: BufferReader): StreamHeaderGroup {
653
+ const subscribeId = reader.readVarInt();
654
+ const trackAlias = reader.readVarInt();
655
+ const groupId = reader.readVarInt();
656
+ const publisherPriority = reader.readUint8();
657
+ return { type: 'stream_header_group', subscribeId, trackAlias, groupId, publisherPriority };
658
+ }
659
+
660
+ function decodeStreamHeaderSubgroup(reader: BufferReader): StreamHeaderSubgroup {
661
+ const subscribeId = reader.readVarInt();
662
+ const trackAlias = reader.readVarInt();
663
+ const groupId = reader.readVarInt();
664
+ const subgroupId = reader.readVarInt();
665
+ const publisherPriority = reader.readUint8();
666
+ return { type: 'stream_header_subgroup', subscribeId, trackAlias, groupId, subgroupId, publisherPriority };
667
+ }
668
+
669
+ // --- Decode dispatch by wire type ID (control messages only) ---
670
+
671
+ type Decoder = (reader: BufferReader) => MoqtMessage;
672
+
673
+ const controlDecoders = new Map<bigint, Decoder>([
674
+ [MESSAGE_TYPE_IDS.client_setup, decodeClientSetup],
675
+ [MESSAGE_TYPE_IDS.server_setup, decodeServerSetup],
676
+ [MESSAGE_TYPE_IDS.subscribe, decodeSubscribe],
677
+ [MESSAGE_TYPE_IDS.subscribe_ok, decodeSubscribeOk],
678
+ [MESSAGE_TYPE_IDS.subscribe_error, decodeSubscribeError],
679
+ [MESSAGE_TYPE_IDS.subscribe_done, decodeSubscribeDone],
680
+ [MESSAGE_TYPE_IDS.subscribe_update, decodeSubscribeUpdate],
681
+ [MESSAGE_TYPE_IDS.unsubscribe, decodeUnsubscribe],
682
+ [MESSAGE_TYPE_IDS.announce, decodeAnnounce],
683
+ [MESSAGE_TYPE_IDS.announce_ok, decodeAnnounceOk],
684
+ [MESSAGE_TYPE_IDS.announce_error, decodeAnnounceError],
685
+ [MESSAGE_TYPE_IDS.announce_cancel, decodeAnnounceCancel],
686
+ [MESSAGE_TYPE_IDS.unannounce, decodeUnannounce],
687
+ [MESSAGE_TYPE_IDS.track_status_request, decodeTrackStatusRequest],
688
+ [MESSAGE_TYPE_IDS.track_status, decodeTrackStatus],
689
+ [MESSAGE_TYPE_IDS.goaway, decodeGoAway],
690
+ [MESSAGE_TYPE_IDS.subscribe_announces, decodeSubscribeAnnounces],
691
+ [MESSAGE_TYPE_IDS.subscribe_announces_ok, decodeSubscribeAnnouncesOk],
692
+ [MESSAGE_TYPE_IDS.subscribe_announces_error, decodeSubscribeAnnouncesError],
693
+ [MESSAGE_TYPE_IDS.unsubscribe_announces, decodeUnsubscribeAnnounces],
694
+ [MESSAGE_TYPE_IDS.max_subscribe_id, decodeMaxSubscribeId],
695
+ [MESSAGE_TYPE_IDS.fetch, decodeFetch],
696
+ [MESSAGE_TYPE_IDS.fetch_ok, decodeFetchOk],
697
+ [MESSAGE_TYPE_IDS.fetch_error, decodeFetchError],
698
+ [MESSAGE_TYPE_IDS.fetch_cancel, decodeFetchCancel],
699
+ ]);
700
+
701
+ // Data stream decoders keyed by wire ID (for disambiguation)
702
+ const dataStreamDecoders = new Map<bigint, Decoder>([
703
+ [MESSAGE_TYPE_IDS.object_stream, decodeObjectStream],
704
+ [MESSAGE_TYPE_IDS.object_datagram, decodeObjectDatagram],
705
+ [MESSAGE_TYPE_IDS.stream_header_track, decodeStreamHeaderTrack],
706
+ [MESSAGE_TYPE_IDS.stream_header_group, decodeStreamHeaderGroup],
707
+ [MESSAGE_TYPE_IDS.stream_header_subgroup, decodeStreamHeaderSubgroup],
708
+ ]);
709
+
710
+ // --- Message type to wire ID mapping ---
711
+ const MESSAGE_TYPE_TO_WIRE: Record<ControlMessageType, bigint> = {
712
+ client_setup: MESSAGE_TYPE_IDS.client_setup,
713
+ server_setup: MESSAGE_TYPE_IDS.server_setup,
714
+ subscribe: MESSAGE_TYPE_IDS.subscribe,
715
+ subscribe_ok: MESSAGE_TYPE_IDS.subscribe_ok,
716
+ subscribe_error: MESSAGE_TYPE_IDS.subscribe_error,
717
+ subscribe_done: MESSAGE_TYPE_IDS.subscribe_done,
718
+ subscribe_update: MESSAGE_TYPE_IDS.subscribe_update,
719
+ unsubscribe: MESSAGE_TYPE_IDS.unsubscribe,
720
+ announce: MESSAGE_TYPE_IDS.announce,
721
+ announce_ok: MESSAGE_TYPE_IDS.announce_ok,
722
+ announce_error: MESSAGE_TYPE_IDS.announce_error,
723
+ announce_cancel: MESSAGE_TYPE_IDS.announce_cancel,
724
+ unannounce: MESSAGE_TYPE_IDS.unannounce,
725
+ track_status_request: MESSAGE_TYPE_IDS.track_status_request,
726
+ track_status: MESSAGE_TYPE_IDS.track_status,
727
+ goaway: MESSAGE_TYPE_IDS.goaway,
728
+ subscribe_announces: MESSAGE_TYPE_IDS.subscribe_announces,
729
+ subscribe_announces_ok: MESSAGE_TYPE_IDS.subscribe_announces_ok,
730
+ subscribe_announces_error: MESSAGE_TYPE_IDS.subscribe_announces_error,
731
+ unsubscribe_announces: MESSAGE_TYPE_IDS.unsubscribe_announces,
732
+ max_subscribe_id: MESSAGE_TYPE_IDS.max_subscribe_id,
733
+ fetch: MESSAGE_TYPE_IDS.fetch,
734
+ fetch_ok: MESSAGE_TYPE_IDS.fetch_ok,
735
+ fetch_error: MESSAGE_TYPE_IDS.fetch_error,
736
+ fetch_cancel: MESSAGE_TYPE_IDS.fetch_cancel,
737
+ };
738
+
739
+ // --- Public codec API ---
740
+
741
+ function encodeMessageImpl(message: MoqtMessage): Uint8Array {
742
+ // Check if it's a data stream message (no type+length framing)
743
+ const dataEncoder = dataStreamEncoders[message.type];
744
+ if (dataEncoder) {
745
+ const writer = new BufferWriter();
746
+ dataEncoder(message as never, writer);
747
+ return writer.finish();
748
+ }
749
+
750
+ // Control message: type + length + payload framing
751
+ const controlEncoder = controlEncoders[message.type as ControlMessageType];
752
+ if (!controlEncoder) {
753
+ throw new Error(`Unknown message type: ${message.type}`);
754
+ }
755
+
756
+ // Encode payload first
757
+ const payloadWriter = new BufferWriter();
758
+ controlEncoder(message as never, payloadWriter);
759
+ const payload = payloadWriter.finish();
760
+
761
+ // Write type + length + payload
762
+ const frameWriter = new BufferWriter();
763
+ frameWriter.writeVarInt(MESSAGE_TYPE_TO_WIRE[message.type as ControlMessageType]);
764
+ frameWriter.writeVarInt(payload.byteLength);
765
+ frameWriter.writeBytes(payload);
766
+ return frameWriter.finish();
767
+ }
768
+
769
+ function decodeMessageImpl(bytes: Uint8Array): DecodeResult<MoqtMessage> {
770
+ try {
771
+ const reader = new BufferReader(bytes, 0);
772
+ const typeId = reader.readVarInt();
773
+
774
+ // Check if this is a data stream type (no length framing)
775
+ if (DATA_STREAM_TYPE_IDS.has(typeId)) {
776
+ const decoder = dataStreamDecoders.get(typeId);
777
+ if (!decoder) {
778
+ return {
779
+ ok: false,
780
+ error: new DecodeError('UNKNOWN_MESSAGE_TYPE', `Unknown data stream type ID: 0x${typeId.toString(16)}`, 0),
781
+ };
782
+ }
783
+ const message = decoder(reader);
784
+ return { ok: true, value: message, bytesRead: reader.offset };
785
+ }
786
+
787
+ // Control message: read length, then decode payload from bounded sub-reader
788
+ const payloadLength = Number(reader.readVarInt());
789
+ const headerBytes = reader.offset; // bytes consumed by type + length
790
+
791
+ if (reader.remaining < payloadLength) {
792
+ return {
793
+ ok: false,
794
+ error: new DecodeError('UNEXPECTED_END', `Not enough bytes for payload: need ${payloadLength}, have ${reader.remaining}`, reader.offset),
795
+ };
796
+ }
797
+
798
+ const payloadBytes = reader.readBytes(payloadLength);
799
+ const totalBytesRead = reader.offset;
800
+
801
+ const decoder = controlDecoders.get(typeId);
802
+ if (!decoder) {
803
+ return {
804
+ ok: false,
805
+ error: new DecodeError('UNKNOWN_MESSAGE_TYPE', `Unknown message type ID: 0x${typeId.toString(16)}`, 0),
806
+ };
807
+ }
808
+
809
+ const payloadReader = new BufferReader(payloadBytes, 0);
810
+ const message = decoder(payloadReader);
811
+ return { ok: true, value: message, bytesRead: totalBytesRead };
812
+ } catch (e) {
813
+ if (e instanceof DecodeError) {
814
+ return { ok: false, error: e };
815
+ }
816
+ throw e;
817
+ }
818
+ }
819
+
820
+ function createStreamDecoderImpl(): TransformStream<Uint8Array, MoqtMessage> {
821
+ let buffer = new Uint8Array(0);
822
+
823
+ return new TransformStream<Uint8Array, MoqtMessage>({
824
+ transform(chunk, controller) {
825
+ // Accumulate incoming data
826
+ const newBuffer = new Uint8Array(buffer.length + chunk.length);
827
+ newBuffer.set(buffer, 0);
828
+ newBuffer.set(chunk, buffer.length);
829
+ buffer = newBuffer;
830
+
831
+ // Try to decode messages from the buffer
832
+ while (buffer.length > 0) {
833
+ const result = decodeMessageImpl(buffer);
834
+ if (!result.ok) {
835
+ if (result.error.code === 'UNEXPECTED_END') {
836
+ // Need more data -- wait for next chunk
837
+ break;
838
+ }
839
+ // Fatal decode error
840
+ controller.error(result.error);
841
+ return;
842
+ }
843
+ controller.enqueue(result.value);
844
+ // Advance the buffer past the consumed bytes
845
+ buffer = buffer.slice(result.bytesRead);
846
+ }
847
+ },
848
+
849
+ flush(controller) {
850
+ // If there is remaining data in the buffer, it is a truncated message
851
+ if (buffer.length > 0) {
852
+ controller.error(
853
+ new DecodeError('UNEXPECTED_END', 'Stream ended with incomplete message data', 0),
854
+ );
855
+ }
856
+ },
857
+ });
858
+ }
859
+
860
+ // --- Factory ---
861
+
862
+ export function createDraft07Codec(): Codec {
863
+ return {
864
+ draft: 'draft-ietf-moq-transport-07',
865
+ encodeMessage: encodeMessageImpl,
866
+ decodeMessage: decodeMessageImpl,
867
+ encodeVarInt,
868
+ decodeVarInt,
869
+ createStreamDecoder: createStreamDecoderImpl,
870
+ };
871
+ }
872
+
873
+ // Export data-stream decoder map for callers that need to disambiguate
874
+ export { dataStreamDecoders };