-
-
Notifications
You must be signed in to change notification settings - Fork 65
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
fix: polyfill W3C compatibility #324
base: master
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,57 +1,60 @@ | ||
/* eslint-disable @typescript-eslint/no-explicit-any */ | ||
import * as exceptions from './Exception'; | ||
import { DataChannel } from '../lib/index'; | ||
import RTCPeerConnection from './RTCPeerConnection'; | ||
import { RTCErrorEvent } from './Events'; | ||
|
||
export default class RTCDataChannel extends EventTarget implements globalThis.RTCDataChannel { | ||
#dataChannel: DataChannel; | ||
#readyState: RTCDataChannelState; | ||
#bufferedAmountLowThreshold: number; | ||
#binaryType: BinaryType; | ||
#bufferedAmountLowThreshold: number = 0; | ||
#binaryType: BinaryType = 'blob'; | ||
#maxPacketLifeTime: number | null; | ||
#maxRetransmits: number | null; | ||
#negotiated: boolean; | ||
#ordered: boolean; | ||
|
||
#closeRequested = false; | ||
#pc: RTCPeerConnection; | ||
|
||
// events | ||
onbufferedamountlow: ((this: RTCDataChannel, ev: Event) => any) | null; | ||
onclose: ((this: RTCDataChannel, ev: Event) => any) | null; | ||
onclosing: ((this: RTCDataChannel, ev: Event) => any) | null; | ||
onerror: ((this: RTCDataChannel, ev: Event) => any) | null; | ||
onmessage: ((this: RTCDataChannel, ev: MessageEvent) => any) | null; | ||
onopen: ((this: RTCDataChannel, ev: Event) => any) | null; | ||
|
||
constructor(dataChannel: DataChannel, opts: globalThis.RTCDataChannelInit = {}) { | ||
onbufferedamountlow: globalThis.RTCDataChannel['onbufferedamountlow']; | ||
onclose: globalThis.RTCDataChannel['onclose']; | ||
onclosing: globalThis.RTCDataChannel['onclosing']; | ||
onerror: globalThis.RTCDataChannel['onerror']; | ||
onmessage: globalThis.RTCDataChannel['onmessage']; | ||
onopen: globalThis.RTCDataChannel['onopen'] | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. bettter to let the types define it, rather than manually declare any, a typed event target implementation would be prefered however |
||
constructor(dataChannel: DataChannel, opts: globalThis.RTCDataChannelInit = {}, pc: RTCPeerConnection) { | ||
super(); | ||
|
||
this.#dataChannel = dataChannel; | ||
this.#binaryType = 'blob'; | ||
this.#readyState = this.#dataChannel.isOpen() ? 'open' : 'connecting'; | ||
this.#bufferedAmountLowThreshold = 0; | ||
this.#maxPacketLifeTime = opts.maxPacketLifeTime || null; | ||
this.#maxRetransmits = opts.maxRetransmits || null; | ||
this.#negotiated = opts.negotiated || false; | ||
this.#ordered = opts.ordered || true; | ||
this.#maxPacketLifeTime = opts.maxPacketLifeTime ?? null; | ||
this.#maxRetransmits = opts.maxRetransmits ?? null; | ||
this.#negotiated = opts.negotiated ?? false; | ||
this.#ordered = opts.ordered ?? true; | ||
this.#pc = pc | ||
|
||
// forward dataChannel events | ||
this.#dataChannel.onOpen(() => { | ||
this.#readyState = 'open'; | ||
this.dispatchEvent(new Event('open', {})); | ||
}); | ||
|
||
this.#dataChannel.onClosed(() => { | ||
// Simulate closing event | ||
if (!this.#closeRequested) { | ||
this.#readyState = 'closing'; | ||
this.dispatchEvent(new Event('closing')); | ||
} | ||
|
||
setImmediate(() => { | ||
this.#readyState = 'closed'; | ||
this.dispatchEvent(new Event('close')); | ||
}); | ||
}); | ||
// we need updated connectionstate, so this is delayed by a single event loop tick | ||
// this is fucked and wonky, needs to be made better | ||
this.#dataChannel.onClosed(() => setTimeout(() => { | ||
if (this.#readyState !== 'closed') { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. the spec is very assine about how and when datachannels should close, this is the closest i was able to bring it inline with said spec, the order of events in peer, ice and dc on closing is important for some libraries |
||
// this should be 'disconnected' but ldc doesn't support that | ||
if (this.#pc.connectionState === 'closed') { | ||
// if the remote connection suddently closes without closing dc first, throw this weird error | ||
this.dispatchEvent(new RTCErrorEvent('error', { error: new RTCError({ errorDetail: 'sctp-failure', sctpCauseCode: 12 }, 'User-Initiated Abort, reason=Close called') })) | ||
} | ||
this.#readyState = 'closing' | ||
this.dispatchEvent(new Event('closing')) | ||
this.#readyState = 'closed' | ||
} | ||
this.dispatchEvent(new Event('close')) | ||
})) | ||
|
||
this.#dataChannel.onError((msg) => { | ||
this.dispatchEvent( | ||
|
@@ -70,16 +73,17 @@ export default class RTCDataChannel extends EventTarget implements globalThis.RT | |
this.dispatchEvent(new Event('bufferedamountlow')); | ||
}); | ||
|
||
this.#dataChannel.onMessage((data) => { | ||
if (ArrayBuffer.isView(data)) { | ||
if (this.binaryType == 'arraybuffer') | ||
data = data.buffer; | ||
else | ||
data = Buffer.from(data.buffer); | ||
this.#dataChannel.onMessage(message => { | ||
let data: Blob | ArrayBufferLike | string | ||
if (!ArrayBuffer.isView(message)) { | ||
data = message | ||
} else if (this.#binaryType === 'blob') { | ||
data = new Blob([message]) | ||
} else { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. don't explicitly use buffer, also cast to Blob when the type requires it, this wasn't even done before |
||
data = message.buffer | ||
} | ||
|
||
this.dispatchEvent(new MessageEvent('message', { data })); | ||
}); | ||
this.dispatchEvent(new MessageEvent('message', { data })) | ||
}) | ||
|
||
// forward events to properties | ||
this.addEventListener('message', (e) => { | ||
|
@@ -89,7 +93,7 @@ export default class RTCDataChannel extends EventTarget implements globalThis.RT | |
if (this.onbufferedamountlow) this.onbufferedamountlow(e); | ||
}); | ||
this.addEventListener('error', (e) => { | ||
if (this.onerror) this.onerror(e); | ||
if (this.onerror) this.onerror(e as RTCErrorEvent); | ||
}); | ||
this.addEventListener('close', (e) => { | ||
if (this.onclose) this.onclose(e); | ||
|
@@ -162,7 +166,11 @@ export default class RTCDataChannel extends EventTarget implements globalThis.RT | |
return this.#readyState; | ||
} | ||
|
||
send(data): void { | ||
get maxMessageSize (): number { | ||
return this.#dataChannel.maxMessageSize() | ||
} | ||
|
||
send(data: string | Blob | ArrayBuffer | ArrayBufferView | Buffer): void { | ||
if (this.#readyState !== 'open') { | ||
throw new exceptions.InvalidStateError( | ||
"Failed to execute 'send' on 'RTCDataChannel': RTCDataChannel.readyState is not 'open'", | ||
|
@@ -171,26 +179,29 @@ export default class RTCDataChannel extends EventTarget implements globalThis.RT | |
|
||
// Needs network error, type error implemented | ||
if (typeof data === 'string') { | ||
if (data.length > this.#dataChannel.maxMessageSize()) throw new TypeError('Max message size exceeded.') | ||
this.#dataChannel.sendMessage(data); | ||
} else if (data instanceof Blob) { | ||
} else if ('arrayBuffer' in data) { | ||
if (data.size > this.#dataChannel.maxMessageSize()) throw new TypeError('Max message size exceeded.') | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. we're not actually interested in blob, but its ab method, this is important for Blob-like libraries |
||
data.arrayBuffer().then((ab) => { | ||
if (process?.versions?.bun) { | ||
this.#dataChannel.sendMessageBinary(Buffer.from(ab)); | ||
} else { | ||
this.#dataChannel.sendMessageBinary(new Uint8Array(ab)); | ||
} | ||
this.#dataChannel.sendMessageBinary( process?.versions?.bun ? Buffer.from(ab) : new Uint8Array(ab)); | ||
}); | ||
} else { | ||
if (process?.versions?.bun) { | ||
this.#dataChannel.sendMessageBinary(Buffer.from(data)); | ||
} else { | ||
this.#dataChannel.sendMessageBinary(new Uint8Array(data)); | ||
} | ||
if (data.byteLength > this.#dataChannel.maxMessageSize()) throw new TypeError('Max message size exceeded.') | ||
this.#dataChannel.sendMessageBinary( process?.versions?.bun ? Buffer.from(data as ArrayBuffer) : new Uint8Array(data as ArrayBuffer)); | ||
} | ||
} | ||
|
||
close(): void { | ||
this.#closeRequested = true; | ||
this.#dataChannel.close(); | ||
close (): void { | ||
this.#readyState = 'closed' | ||
setTimeout(() => { | ||
if (this.#pc.connectionState === 'closed') { | ||
// if the remote connection suddently closes without closing dc first, throw this weird error | ||
// can this be done better? | ||
this.dispatchEvent(new RTCErrorEvent('error', { error: new RTCError({ errorDetail: 'sctp-failure', sctpCauseCode: 12 }, 'User-Initiated Abort, reason=Close called') })) | ||
} | ||
}) | ||
|
||
this.#dataChannel.close() | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,28 +1,24 @@ | ||
/* eslint-disable @typescript-eslint/no-explicit-any */ | ||
import RTCIceTransport from './RTCIceTransport'; | ||
import RTCPeerConnection from './RTCPeerConnection'; | ||
|
||
export default class RTCDtlsTransport extends EventTarget implements globalThis.RTCDtlsTransport { | ||
#pc: RTCPeerConnection = null; | ||
#iceTransport = null; | ||
|
||
onstatechange: ((this: RTCDtlsTransport, ev: Event) => any) | null = null; | ||
onerror: ((this: RTCDtlsTransport, ev: Event) => any) | null = null; | ||
onstatechange: globalThis.RTCDtlsTransport['onstatechange']; | ||
onerror: globalThis.RTCDtlsTransport['onstatechange']; | ||
|
||
constructor(init: { pc: RTCPeerConnection, extraFunctions }) { | ||
constructor({ pc }: { pc: RTCPeerConnection }) { | ||
super(); | ||
this.#pc = init.pc; | ||
this.#pc = pc; | ||
|
||
this.#iceTransport = new RTCIceTransport({ pc: init.pc, extraFunctions: init.extraFunctions }); | ||
this.#iceTransport = new RTCIceTransport({ pc }); | ||
|
||
// forward peerConnection events | ||
this.#pc.addEventListener('connectionstatechange', () => { | ||
this.dispatchEvent(new Event('statechange')); | ||
}); | ||
|
||
// forward events to properties | ||
this.addEventListener('statechange', (e) => { | ||
if (this.onstatechange) this.onstatechange(e); | ||
const e = new Event('statechange'); | ||
this.dispatchEvent(e); | ||
this.onstatechange?.(e); | ||
}); | ||
} | ||
|
||
|
@@ -33,15 +29,12 @@ export default class RTCDtlsTransport extends EventTarget implements globalThis. | |
get state(): RTCDtlsTransportState { | ||
// reduce state from new, connecting, connected, disconnected, failed, closed, unknown | ||
// to RTCDtlsTRansport states new, connecting, connected, closed, failed | ||
let state = this.#pc ? this.#pc.connectionState : 'new'; | ||
if (state === 'disconnected') { | ||
state = 'closed'; | ||
} | ||
return state; | ||
if (this.#pc.connectionState === 'disconnected') return 'closed' | ||
return this.#pc.connectionState | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. pc is always defined |
||
} | ||
|
||
getRemoteCertificates(): ArrayBuffer[] { | ||
// TODO: implement | ||
// TODO: implement, not supported by all browsers anyways | ||
return [new ArrayBuffer(0)]; | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
required by spec