Skip to content
This repository has been archived by the owner on Mar 1, 2024. It is now read-only.

Add TCP+Relay detection event #485

Merged
merged 22 commits into from
Feb 19, 2024
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
Show all changes
22 commits
Select commit Hold shift + click to select a range
8b5bbd8
Change the label for the remote candidate
david-macpherson Feb 13, 2024
daff998
Added the following missing properties id,timestamp,type,lastPacketRe…
david-macpherson Feb 13, 2024
5542d91
Added relayProtocol and transport ID to the candidate Stat
david-macpherson Feb 13, 2024
1be9ed5
Added relayProtocol and transport ID to the candidate Stat
david-macpherson Feb 13, 2024
03ee5b6
Set the parsed candidate pair stat to the nominated and selected pair
david-macpherson Feb 13, 2024
7d97eb1
Implemeneted a stream waring event
david-macpherson Feb 14, 2024
f05ce78
Implemented the emitting of the stream warning event when if the loca…
david-macpherson Feb 14, 2024
5b9f8b0
Fixed up styling
david-macpherson Feb 14, 2024
1b8e9c3
Refactored the candidate pair to an array, Implemneted a helper funct…
david-macpherson Feb 15, 2024
86fbdb2
Refactored to use Array.find
david-macpherson Feb 15, 2024
0e89638
Replaced the stream warning event with WebRtcTCPRelayDetectedEvent
david-macpherson Feb 15, 2024
bdf41af
Implemeneted the WebRtcTCPRelayDetectedEvent to dispatch if the web r…
david-macpherson Feb 15, 2024
55d02d6
Refactored the stats panel to use the getActiveCandidatePair
david-macpherson Feb 15, 2024
fb6bf16
Refactored the webrtc tcp detect check to act on an event instead of …
david-macpherson Feb 15, 2024
533d88a
Refactored the getting of the active candidate pair to be stored as a…
david-macpherson Feb 15, 2024
9d26239
Updated the candidate pair to the candidate pairs array
david-macpherson Feb 15, 2024
7eab39d
Added a warning if the stream is relayed over tcp
david-macpherson Feb 15, 2024
51babfd
Moved the logic for emitting the webrtc tcp relay detect event to it'…
david-macpherson Feb 15, 2024
d00bf75
Refactored the binding of the stats recieved event to the setup web r…
david-macpherson Feb 16, 2024
8567df6
Removed un needed new lines
david-macpherson Feb 16, 2024
15ed4ad
Added a check if the get active candidate is null to return a new can…
david-macpherson Feb 16, 2024
ad24cc9
Reverted the refactor the binding of the stats recieved event function
david-macpherson Feb 19, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
20 changes: 14 additions & 6 deletions Frontend/library/src/PeerConnectionController/AggregatedStats.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ export class AggregatedStats {
inboundAudioStats: InboundAudioStats;
lastVideoStats: InboundVideoStats;
lastAudioStats: InboundAudioStats;
candidatePair: CandidatePairStats;
candidatePairs: Array<CandidatePairStats>;
DataChannelStats: DataChannelStats;
localCandidates: Array<CandidateStat>;
remoteCandidates: Array<CandidateStat>;
Expand All @@ -37,7 +37,6 @@ export class AggregatedStats {
constructor() {
this.inboundVideoStats = new InboundVideoStats();
this.inboundAudioStats = new InboundAudioStats();
this.candidatePair = new CandidatePairStats();
this.DataChannelStats = new DataChannelStats();
this.outBoundVideoStats = new OutBoundVideoStats();
this.sessionStats = new SessionStats();
Expand All @@ -52,6 +51,7 @@ export class AggregatedStats {
processStats(rtcStatsReport: RTCStatsReport) {
this.localCandidates = new Array<CandidateStat>();
this.remoteCandidates = new Array<CandidateStat>();
this.candidatePairs = new Array<CandidatePairStats>();

rtcStatsReport.forEach((stat) => {
const type: RTCStatsTypePS = stat.type;
Expand Down Expand Up @@ -120,10 +120,10 @@ export class AggregatedStats {
* @param stat - the stats coming in from ice candidates
*/
handleCandidatePair(stat: CandidatePairStats) {
// If the candidate pair has received bytes then set as the candidate pair
if (stat.bytesReceived > 0){
this.candidatePair = stat;
}

// Add the candidate pair to the candidate pair array
this.candidatePairs.push(stat)

}

/**
Expand Down Expand Up @@ -306,4 +306,12 @@ export class AggregatedStats {
isNumber(value: unknown): boolean {
return typeof value === 'number' && isFinite(value);
}

/**
* Helper function to return the active candidate pair
* @returns The candidate pair that is currently receiving data
*/
public getActiveCandidatePair(): CandidatePairStats | null {
return this.candidatePairs.find((candidatePair) => candidatePair.bytesReceived > 0, null)
}
lukehb marked this conversation as resolved.
Show resolved Hide resolved
}
48 changes: 26 additions & 22 deletions Frontend/library/src/PixelStreaming/PixelStreaming.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@ import {
StreamPreConnectEvent,
StreamReconnectEvent,
StreamPreDisconnectEvent,
StreamWarningEvent,
VideoEncoderAvgQPEvent,
VideoInitializedEvent,
WebRtcAutoConnectEvent,
Expand All @@ -29,7 +28,8 @@ import {
WebRtcSdpEvent,
DataChannelLatencyTestResponseEvent,
DataChannelLatencyTestResultEvent,
PlayerCountEvent
PlayerCountEvent,
WebRtcTCPRelayDetectedEvent
} from '../Util/EventEmitter';
import { MessageOnScreenKeyboard } from '../WebSockets/MessageReceive';
import { WebXRController } from '../WebXR/WebXRController';
Expand Down Expand Up @@ -80,13 +80,17 @@ export class PixelStreaming {

private _eventEmitter: EventEmitter;

private _webRtcTCPRelayDetectedCheck: boolean;


/**
* @param config - A newly instantiated config object
* @param overrides - Parameters to override default behaviour
* returns the base Pixel streaming object
*/
constructor(config: Config, overrides?: PixelStreamingOverrides) {
this.config = config;
this._webRtcTCPRelayDetectedCheck = false;

if (overrides?.videoElementParent) {
this._videoElementParent = overrides.videoElementParent;
Expand Down Expand Up @@ -535,28 +539,28 @@ export class PixelStreaming {
new StatsReceivedEvent({ aggregatedStats: videoStats })
);

lukehb marked this conversation as resolved.
Show resolved Hide resolved
// Get the local candidate from the candidate pair
let localSelectedCandidate = videoStats.localCandidates.find((candid) => {
return candid.id == videoStats.candidatePair.localCandidateId
})

// Check if the local candidate is relaying over TCP
if (localSelectedCandidate.candidateType == 'relay' && localSelectedCandidate.relayProtocol == 'tcp'){
// Check if the web rtc rely detected check has completed
if (!this._webRtcTCPRelayDetectedCheck){
lukehb marked this conversation as resolved.
Show resolved Hide resolved

// Send a warning to the logger informing the user the stream will be severely degraded
Logger.Warning(
Logger.GetStackTrace(),
`Stream quality severely degraded, local connection is relayed over TCP due to the local network environment.`
);
// Get the active candidate pair
let activeCandidatePair = videoStats.getActiveCandidatePair();

// Emit a stream warning event
this._eventEmitter.dispatchEvent(
new StreamWarningEvent({
candidateType: localSelectedCandidate.candidateType,
protocol: localSelectedCandidate.protocol,
relayProtocol: localSelectedCandidate.relayProtocol,
})
);
// Check if the active candidate pair is not null
if (activeCandidatePair != null){

// Get the local candidate assigned to the active candidate pair
let localCandidate = videoStats.localCandidates.find((candidate) => candidate.id == activeCandidatePair.localCandidateId,null)

// Check if the local candidate is not null, candidate type is relay and the relay protocol is tcp
if (localCandidate != null && localCandidate.candidateType == 'relay' && localCandidate.relayProtocol == 'tcp'){

// Send the web rtc tcp relay detected Evebt
this._eventEmitter.dispatchEvent(new WebRtcTCPRelayDetectedEvent());
}

// set the web rtc tcp relay detected check to true
this._webRtcTCPRelayDetectedCheck = true;
}
}
}

Expand Down
21 changes: 7 additions & 14 deletions Frontend/library/src/Util/EventEmitter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -538,19 +538,12 @@ export class PlayerCountEvent extends Event {
}

/**
* An event that is emitted when the stream quality is degraded due to local network environment.
* An event that is emitted when the webRTC connections is relayed over TCP.
*/
export class StreamWarningEvent extends Event {
readonly type: 'streamWarning';
readonly data: {
/** stream warning event */
protocol: 'tcp' | 'udp'
relayProtocol: 'tcp' | 'udp' | 'tls'
candidateType: string
};
constructor(data: StreamWarningEvent['data']) {
super('streamWarning');
this.data = data;
export class WebRtcTCPRelayDetectedEvent extends Event {
readonly type: 'webRtcTCPRelayDetected';
constructor() {
super('webRtcTCPRelayDetected');
}
}

Expand All @@ -574,7 +567,6 @@ export type PixelStreamingEvent =
| StreamPreConnectEvent
| StreamReconnectEvent
| StreamPreDisconnectEvent
| StreamWarningEvent
| PlayStreamErrorEvent
| PlayStreamEvent
| PlayStreamRejectedEvent
Expand All @@ -591,7 +583,8 @@ export type PixelStreamingEvent =
| XrSessionStartedEvent
| XrSessionEndedEvent
| XrFrameEvent
| PlayerCountEvent;
| PlayerCountEvent
| WebRtcTCPRelayDetectedEvent;

export class EventEmitter extends EventTarget {
/**
Expand Down
6 changes: 3 additions & 3 deletions Frontend/ui-library/src/UI/StatsPanel.ts
Original file line number Diff line number Diff line change
Expand Up @@ -321,11 +321,11 @@ export class StatsPanel {
// RTT
const netRTT =
Object.prototype.hasOwnProperty.call(
lukehb marked this conversation as resolved.
Show resolved Hide resolved
stats.candidatePair,
stats.getActiveCandidatePair(),
'currentRoundTripTime'
) && stats.isNumber(stats.candidatePair.currentRoundTripTime)
) && stats.isNumber(stats.getActiveCandidatePair().currentRoundTripTime)
? numberFormat.format(
stats.candidatePair.currentRoundTripTime * 1000
stats.getActiveCandidatePair().currentRoundTripTime * 1000
)
: "Can't calculate";
this.addOrUpdateStat('RTTStat', 'Net RTT (ms)', netRTT);
Expand Down
Loading