import * as Message from "./message" import * as Stream from "../stream" import * as MP4 from "../mp4" import Audio from "../audio" import Video from "../video" // @ts-ignore bundler embeds data import fingerprint from 'bundle-text:./fingerprint.hex'; /// export interface PlayerInit { url: string; canvas: HTMLCanvasElement; } export class Player { quic: Promise; api: Promise; tracks: Map audio: Audio; video: Video; constructor(props: PlayerInit) { this.tracks = new Map(); // TODO move these to another class so this only deals with the transport. this.audio = new Audio({}) this.video = new Video({ canvas: props.canvas.transferControlToOffscreen(), }) this.quic = this.connect(props.url) // Create a unidirectional stream for all of our messages this.api = this.quic.then((q) => { return q.createUnidirectionalStream() }) // async functions this.receiveStreams() } async close() { (await this.quic).close() } async connect(url: string): Promise { // Convert the hex to binary. let hash = []; for (let c = 0; c < fingerprint.length-1; c += 2) { hash.push(parseInt(fingerprint.substring(c, c+2), 16)); } const quic = new WebTransport(url, { "serverCertificateHashes": [{ "algorithm": "sha-256", "value": new Uint8Array(hash), }] }) await quic.ready return quic } async sendMessage(msg: any) { const payload = JSON.stringify(msg) const size = payload.length + 8 const stream = await this.api const writer = new Stream.Writer(stream) await writer.uint32(size) await writer.string("warp") await writer.string(payload) writer.release() } async receiveStreams() { const q = await this.quic const streams = q.incomingUnidirectionalStreams.getReader() while (true) { const result = await streams.read() if (result.done) break const stream = result.value this.handleStream(stream) // don't await } } async handleStream(stream: ReadableStream) { let r = new Stream.Reader(stream) while (!await r.done()) { const size = await r.uint32(); const typ = new TextDecoder('utf-8').decode(await r.bytes(4)); if (typ != "warp") throw "expected warp atom" if (size < 8) throw "atom too small" const payload = new TextDecoder('utf-8').decode(await r.bytes(size - 8)); const msg = JSON.parse(payload) if (msg.init) { return this.handleInit(r, msg.init as Message.Init) } else if (msg.segment) { return this.handleSegment(r, msg.segment as Message.Segment) } } } async handleInit(stream: Stream.Reader, msg: Message.Init) { let track = this.tracks.get(msg.id); if (!track) { track = new MP4.InitParser() this.tracks.set(msg.id, track) } while (1) { const data = await stream.read() if (!data) break track.push(data) } const info = await track.info if (info.audioTracks.length + info.videoTracks.length != 1) { throw new Error("expected a single track") } if (info.audioTracks) { this.audio.init({ track: msg.id, info: info, raw: track.raw, }) } else if (info.videoTracks) { this.video.init({ track: msg.id, info: info, raw: track.raw, }) } else { throw new Error("init is neither audio nor video") } } async handleSegment(stream: Stream.Reader, msg: Message.Segment) { let track = this.tracks.get(msg.init); if (!track) { track = new MP4.InitParser() this.tracks.set(msg.init, track) } // Wait until we learn if this is an audio or video track const info = await track.info if (info.audioTracks) { this.audio.segment({ track: msg.init, buffer: stream.buffer, reader: stream.reader, }) } else if (info.videoTracks) { this.video.segment({ track: msg.init, buffer: stream.buffer, reader: stream.reader, }) } else { throw new Error("segment is neither audio nor video") } } }