@lumencast/runtime 0.9.0 → 0.10.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 (90) hide show
  1. package/dist/.tsbuildinfo +1 -1
  2. package/dist/app.d.ts +6 -1
  3. package/dist/app.d.ts.map +1 -1
  4. package/dist/app.js +3 -1
  5. package/dist/app.js.map +1 -1
  6. package/dist/{broadcast-ryjLRD5q.js → broadcast-L5wm2I6J.js} +3 -3
  7. package/dist/{broadcast-ryjLRD5q.js.map → broadcast-L5wm2I6J.js.map} +1 -1
  8. package/dist/{control-AgxbXOVS.js → control-eEUG7unp.js} +4 -4
  9. package/dist/{control-AgxbXOVS.js.map → control-eEUG7unp.js.map} +1 -1
  10. package/dist/index-Clrya_9l.js +1281 -0
  11. package/dist/index-Clrya_9l.js.map +1 -0
  12. package/dist/index.d.ts +2 -0
  13. package/dist/index.d.ts.map +1 -1
  14. package/dist/index.html +1 -1
  15. package/dist/index.js +11 -0
  16. package/dist/index.js.map +1 -1
  17. package/dist/lumencast.js +18 -13
  18. package/dist/mount.d.ts.map +1 -1
  19. package/dist/mount.js +11 -0
  20. package/dist/mount.js.map +1 -1
  21. package/dist/overlay/runtime-context.d.ts +10 -0
  22. package/dist/overlay/runtime-context.d.ts.map +1 -1
  23. package/dist/overlay/runtime-context.js.map +1 -1
  24. package/dist/render/bundle.d.ts +1 -1
  25. package/dist/render/bundle.d.ts.map +1 -1
  26. package/dist/render/bundle.js.map +1 -1
  27. package/dist/render/primitives/capture.d.ts +13 -4
  28. package/dist/render/primitives/capture.d.ts.map +1 -1
  29. package/dist/render/primitives/capture.js +54 -22
  30. package/dist/render/primitives/capture.js.map +1 -1
  31. package/dist/render/primitives/index.d.ts.map +1 -1
  32. package/dist/render/primitives/index.js +4 -0
  33. package/dist/render/primitives/index.js.map +1 -1
  34. package/dist/render/primitives/live-peer-video.d.ts +27 -0
  35. package/dist/render/primitives/live-peer-video.d.ts.map +1 -0
  36. package/dist/render/primitives/live-peer-video.js +64 -0
  37. package/dist/render/primitives/live-peer-video.js.map +1 -0
  38. package/dist/render/primitives/media.d.ts +37 -12
  39. package/dist/render/primitives/media.d.ts.map +1 -1
  40. package/dist/render/primitives/media.js +43 -17
  41. package/dist/render/primitives/media.js.map +1 -1
  42. package/dist/render/primitives/meet-peer.d.ts +31 -0
  43. package/dist/render/primitives/meet-peer.d.ts.map +1 -0
  44. package/dist/render/primitives/meet-peer.js +46 -0
  45. package/dist/render/primitives/meet-peer.js.map +1 -0
  46. package/dist/render/prop-allowlist.d.ts.map +1 -1
  47. package/dist/render/prop-allowlist.js +27 -1
  48. package/dist/render/prop-allowlist.js.map +1 -1
  49. package/dist/render/tree.js +42 -8
  50. package/dist/render/tree.js.map +1 -1
  51. package/dist/{status-pill-BxCdj-KZ.js → status-pill-elORkMrh.js} +2 -2
  52. package/dist/{status-pill-BxCdj-KZ.js.map → status-pill-elORkMrh.js.map} +1 -1
  53. package/dist/{test-CaRHj_J6.js → test-7q_KJkdX.js} +4 -4
  54. package/dist/{test-CaRHj_J6.js.map → test-7q_KJkdX.js.map} +1 -1
  55. package/dist/{tree-BLIxJbD3.js → tree-BMxx5170.js} +522 -436
  56. package/dist/tree-BMxx5170.js.map +1 -0
  57. package/dist/types.d.ts +13 -0
  58. package/dist/types.d.ts.map +1 -1
  59. package/dist/webrtc/index.d.ts +76 -0
  60. package/dist/webrtc/index.d.ts.map +1 -0
  61. package/dist/webrtc/index.js +180 -0
  62. package/dist/webrtc/index.js.map +1 -0
  63. package/dist/webrtc/meet-viewer.d.ts +139 -0
  64. package/dist/webrtc/meet-viewer.d.ts.map +1 -0
  65. package/dist/webrtc/meet-viewer.js +379 -0
  66. package/dist/webrtc/meet-viewer.js.map +1 -0
  67. package/dist/webrtc/peer-stream-registry.d.ts +21 -0
  68. package/dist/webrtc/peer-stream-registry.d.ts.map +1 -0
  69. package/dist/webrtc/peer-stream-registry.js +77 -0
  70. package/dist/webrtc/peer-stream-registry.js.map +1 -0
  71. package/package.json +4 -4
  72. package/src/app.tsx +9 -0
  73. package/src/index.ts +35 -0
  74. package/src/mount.ts +11 -0
  75. package/src/overlay/runtime-context.tsx +10 -0
  76. package/src/render/bundle.ts +11 -1
  77. package/src/render/primitives/capture.tsx +73 -28
  78. package/src/render/primitives/index.ts +4 -0
  79. package/src/render/primitives/live-peer-video.tsx +90 -0
  80. package/src/render/primitives/media.tsx +66 -17
  81. package/src/render/primitives/meet-peer.tsx +57 -0
  82. package/src/render/prop-allowlist.ts +27 -1
  83. package/src/render/tree.tsx +44 -8
  84. package/src/types.ts +13 -0
  85. package/src/webrtc/index.ts +252 -0
  86. package/src/webrtc/meet-viewer.ts +497 -0
  87. package/src/webrtc/peer-stream-registry.ts +93 -0
  88. package/dist/index-DrXsLYhe.js +0 -903
  89. package/dist/index-DrXsLYhe.js.map +0 -1
  90. package/dist/tree-BLIxJbD3.js.map +0 -1
@@ -0,0 +1,252 @@
1
+ // WebRTC viewer — public surface (ADR 006 §3.3, issue #3).
2
+ //
3
+ // `createPeerViewer` wires a `MeetViewer` (mesh, viewer role) to a
4
+ // `PeerStreamRegistry` and returns the `resolvePeerStream` / `subscribePeerStream`
5
+ // the `media` primitive's LIVE mode (#4) consumes via `MountOptions`. This is
6
+ // the #3↔#4 bridge : the viewer receives peers, the registry maps
7
+ // `peer_label → MediaStream`, the primitive renders it in `<video srcObject>`.
8
+ //
9
+ // MULTI-ROOM (final model) : `createMultiRoomPeerViewer({ rooms: [...] })` joins
10
+ // EVERY room with its own mesh and feeds ONE shared registry, so the `meet.peer`
11
+ // renderer resolves a `peer_label` to its stream whatever room it came from.
12
+
13
+ import { MeetViewer, type MeetViewerOptions, type RemoteTrackEvent } from "./meet-viewer.js";
14
+ import {
15
+ createPeerStreamRegistry,
16
+ type PeerStreamListener,
17
+ type PeerStreamRegistry,
18
+ } from "./peer-stream-registry.js";
19
+
20
+ export {
21
+ MeetViewer,
22
+ type MeetViewerOptions,
23
+ type MeetViewerDeps,
24
+ type PeerInfo,
25
+ type RemoteTrackEvent,
26
+ } from "./meet-viewer.js";
27
+ export {
28
+ createPeerStreamRegistry,
29
+ type PeerStreamRegistry,
30
+ type PeerStreamListener,
31
+ } from "./peer-stream-registry.js";
32
+
33
+ export interface PeerViewer {
34
+ /** Join the room(s) (viewer role, no capture). */
35
+ join: () => Promise<void>;
36
+ /** Leave + tear down all peer connections (the track owner releases here). */
37
+ leave: () => void;
38
+ /** #4 contract — pass to `mount({ resolvePeerStream })`. Synchronous. */
39
+ resolvePeerStream: (peerLabel: string) => MediaStream | null;
40
+ /** Push channel for the LIVE primitive to re-render on connect/disconnect. */
41
+ subscribePeerStream: (peerLabel: string, listener: PeerStreamListener) => () => void;
42
+ /** The underlying registry, for advanced hosts. */
43
+ registry: PeerStreamRegistry;
44
+ /** Single-room only — the underlying mesh, for diagnostics. Absent in the
45
+ * multi-room viewer (it owns N meshes). */
46
+ viewer?: MeetViewer;
47
+ }
48
+
49
+ /** Feed a (possibly shared) registry from ONE viewer's lifecycle. A label is
50
+ * owned by the FIRST room that connects it (`label-collision` policy) : `claim`
51
+ * decides whether this viewer may publish/withdraw a given label, so a second
52
+ * room carrying the same `peer_label` never clobbers the first. Returns the
53
+ * viewer + a `dispose` that closes the mesh and releases this viewer's labels. */
54
+ /** Normalise a peer name / peer_label into the SAME key both sides agree on.
55
+ * The publisher announces a FREE name on the mesh (e.g. "Publisher 366"), but
56
+ * the scene's `peer_label` is slugified at LSML export (from-scene → "publisher
57
+ * _366", per source-node.ts `slugifyToLabel`). Strict `peerName === peer_label`
58
+ * therefore never matches. Applying the SAME slug to BOTH the indexing side
59
+ * (peerName) and the resolution side (peer_label) makes the map work regardless
60
+ * of format (a raw label and its slug collapse to the same key). Mirrors
61
+ * Prism `slugifyToLabel`. */
62
+ export function labelKey(s: string): string {
63
+ return s
64
+ .toLowerCase()
65
+ .replace(/[^a-z0-9_-]+/g, "_")
66
+ .replace(/^[_-]+|[_-]+$/g, "");
67
+ }
68
+
69
+ function wireViewer(
70
+ viewer: MeetViewer,
71
+ registry: PeerStreamRegistry,
72
+ claim: {
73
+ acquire: (label: string, viewer: MeetViewer) => boolean;
74
+ release: (label: string, viewer: MeetViewer) => void;
75
+ },
76
+ ): void {
77
+ // Index the registry by the NORMALISED label (slug of peerName), so it matches
78
+ // the scene node's slugified `peer_label`. Never the opaque `peerId`.
79
+ viewer.on("remote-track", (e: RemoteTrackEvent) => {
80
+ const key = labelKey(e.peerName);
81
+ if (claim.acquire(key, viewer)) registry.set(key, e.stream);
82
+ });
83
+ viewer.on("peer-left", (e) => {
84
+ const key = labelKey(e.peerName);
85
+ if (claim.acquire(key, viewer)) {
86
+ registry.remove(key);
87
+ claim.release(key, viewer);
88
+ }
89
+ });
90
+ }
91
+
92
+ /** Single-room ownership : trivially, this lone viewer owns every label. */
93
+ const SOLE_OWNER = {
94
+ acquire: () => true,
95
+ release: () => {},
96
+ };
97
+
98
+ /** Build a viewer + registry for a SINGLE room and expose the #4 resolver.
99
+ * (Back-compat surface — `createMultiRoomPeerViewer` is the final model.) */
100
+ export function createPeerViewer(options: MeetViewerOptions): PeerViewer {
101
+ const registry = createPeerStreamRegistry();
102
+ const viewer = new MeetViewer(options);
103
+ wireViewer(viewer, registry, SOLE_OWNER);
104
+ return {
105
+ join: () => viewer.join(),
106
+ leave: () => {
107
+ viewer.leave();
108
+ registry.clear();
109
+ },
110
+ resolvePeerStream: (peerLabel) => registry.resolve(labelKey(peerLabel)),
111
+ subscribePeerStream: (peerLabel, listener) => registry.subscribe(labelKey(peerLabel), listener),
112
+ registry,
113
+ viewer,
114
+ };
115
+ }
116
+
117
+ /** A single room's connection params (one mesh per entry). */
118
+ export type RoomOptions = Omit<MeetViewerOptions, "name"> & {
119
+ /** This viewer's announce name on the mesh. Defaults to "solar-viewer". */
120
+ name?: string;
121
+ };
122
+
123
+ export interface MultiRoomPeerViewerOptions {
124
+ rooms: RoomOptions[];
125
+ /** Shared deps (WS/RTCPeerConnection/MediaStream) applied to every room when a
126
+ * room entry does not override them — used to test without a browser stack. */
127
+ deps?: MeetViewerOptions["deps"];
128
+ }
129
+
130
+ export interface MultiRoomPeerViewer extends PeerViewer {
131
+ /** Reconcile the live room set : open meshes for new rooms, close meshes for
132
+ * removed ones (matched by `roomId`). Best-effort ; idempotent for an
133
+ * unchanged set. Newly opened rooms are joined immediately. */
134
+ setRooms: (rooms: RoomOptions[]) => Promise<void>;
135
+ }
136
+
137
+ /** The FINAL viewer model : join N rooms, aggregate every peer into ONE shared
138
+ * registry keyed by `peer_label`. The `meet.peer` renderer resolves a label to
139
+ * its stream regardless of which room the peer published in.
140
+ *
141
+ * LABEL COLLISION (improbable at the POC) : FIRST-CONNECTED-WINS. The room that
142
+ * first connects a `peer_label` owns it ; a second room carrying the same label
143
+ * is ignored until the owner releases it (on the owner's `peer-left`). This is
144
+ * deterministic and never flips a live source under the compositor.
145
+ *
146
+ * LIFECYCLE : one mesh per room ; `leave()` closes all. `setRooms()` reconciles
147
+ * on re-arm (close removed rooms, open new ones), matched by `roomId`. */
148
+ export function createMultiRoomPeerViewer(
149
+ options: MultiRoomPeerViewerOptions,
150
+ ): MultiRoomPeerViewer {
151
+ const registry = createPeerStreamRegistry();
152
+ // roomId → { viewer, joined }
153
+ const meshes = new Map<string, { viewer: MeetViewer }>();
154
+ // peer_label → owning viewer (first-connected-wins).
155
+ const owners = new Map<string, MeetViewer>();
156
+
157
+ const claim = {
158
+ acquire: (label: string, viewer: MeetViewer): boolean => {
159
+ const owner = owners.get(label);
160
+ if (owner === undefined) {
161
+ owners.set(label, viewer);
162
+ return true;
163
+ }
164
+ return owner === viewer; // only the owning room may publish/withdraw
165
+ },
166
+ release: (label: string, viewer: MeetViewer): void => {
167
+ if (owners.get(label) === viewer) owners.delete(label);
168
+ },
169
+ };
170
+
171
+ function openRoom(room: RoomOptions): void {
172
+ if (meshes.has(room.roomId)) return; // idempotent
173
+ const viewer = new MeetViewer({
174
+ name: room.name ?? "solar-viewer",
175
+ ...room,
176
+ ...(options.deps !== undefined && room.deps === undefined ? { deps: options.deps } : {}),
177
+ });
178
+ wireViewer(viewer, registry, claim);
179
+ meshes.set(room.roomId, { viewer });
180
+ }
181
+
182
+ function closeRoom(roomId: string): void {
183
+ const mesh = meshes.get(roomId);
184
+ if (mesh === undefined) return;
185
+ // Release every label this viewer owns BEFORE closing, so a surviving room
186
+ // can take over the label and the registry drops the gone stream.
187
+ for (const [label, owner] of [...owners.entries()]) {
188
+ if (owner === mesh.viewer) {
189
+ registry.remove(label);
190
+ owners.delete(label);
191
+ }
192
+ }
193
+ mesh.viewer.leave();
194
+ meshes.delete(roomId);
195
+ }
196
+
197
+ for (const room of options.rooms) openRoom(room);
198
+
199
+ return {
200
+ join: async () => {
201
+ await Promise.all([...meshes.values()].map((m) => m.viewer.join()));
202
+ },
203
+ leave: () => {
204
+ for (const roomId of [...meshes.keys()]) closeRoom(roomId);
205
+ registry.clear();
206
+ },
207
+ setRooms: async (rooms) => {
208
+ const next = new Set(rooms.map((r) => r.roomId));
209
+ // Close rooms no longer present.
210
+ for (const roomId of [...meshes.keys()]) {
211
+ if (!next.has(roomId)) closeRoom(roomId);
212
+ }
213
+ // Open + join rooms newly added.
214
+ const added: MeetViewer[] = [];
215
+ for (const room of rooms) {
216
+ if (!meshes.has(room.roomId)) {
217
+ openRoom(room);
218
+ const m = meshes.get(room.roomId);
219
+ if (m) added.push(m.viewer);
220
+ }
221
+ }
222
+ await Promise.all(added.map((v) => v.join()));
223
+ },
224
+ resolvePeerStream: (peerLabel) => registry.resolve(labelKey(peerLabel)),
225
+ subscribePeerStream: (peerLabel, listener) => registry.subscribe(labelKey(peerLabel), listener),
226
+ registry,
227
+ };
228
+ }
229
+
230
+ /** The shape the Prism host injects on `window.__ZAB_PEER_VIEWER__`. The FINAL
231
+ * model is the multi-room `{ rooms: [...] }` ; the legacy single-room shape is
232
+ * still accepted for back-compat (treated as a one-room array). */
233
+ export type PeerViewerInjection =
234
+ | MultiRoomPeerViewerOptions
235
+ | (Omit<MeetViewerOptions, "name"> & { name?: string });
236
+
237
+ /** Normalise either injection shape into a multi-room viewer. forge-prism passes
238
+ * `window.__ZAB_PEER_VIEWER__` straight through ; a bare single-room object is
239
+ * wrapped as `{ rooms: [it] }`. */
240
+ export function createPeerViewerFromInjection(injection: PeerViewerInjection): MultiRoomPeerViewer {
241
+ if ("rooms" in injection && Array.isArray(injection.rooms)) {
242
+ return createMultiRoomPeerViewer(injection);
243
+ }
244
+ // Legacy single-room shape → one-room array.
245
+ const { name, deps, ...room } = injection as Omit<MeetViewerOptions, "name"> & {
246
+ name?: string;
247
+ };
248
+ return createMultiRoomPeerViewer({
249
+ rooms: [{ ...room, ...(name !== undefined ? { name } : {}) }],
250
+ ...(deps !== undefined ? { deps } : {}),
251
+ });
252
+ }