-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathOutboundNetworkIssueDetector.ts
112 lines (91 loc) · 3.46 KB
/
OutboundNetworkIssueDetector.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
import {
IssueDetectorResult,
IssueReason,
IssueType,
WebRTCStatsParsed,
} from '../types';
import BaseIssueDetector, { BaseIssueDetectorParams } from './BaseIssueDetector';
interface OutboundNetworkIssueDetectorParams extends BaseIssueDetectorParams {
highPacketLossThresholdPct?: number;
highJitterThreshold?: number;
}
class OutboundNetworkIssueDetector extends BaseIssueDetector {
readonly #highPacketLossThresholdPct: number;
readonly #highJitterThreshold: number;
constructor(params: OutboundNetworkIssueDetectorParams = {}) {
super();
this.#highPacketLossThresholdPct = params.highPacketLossThresholdPct ?? 5;
this.#highJitterThreshold = params.highJitterThreshold ?? 200;
}
performDetection(data: WebRTCStatsParsed): IssueDetectorResult {
return this.processData(data);
}
private processData(data: WebRTCStatsParsed): IssueDetectorResult {
const issues: IssueDetectorResult = [];
const remoteInboundRTPStreamsStats = [
...data.remote?.audio.inbound || [],
...data.remote?.video.inbound || [],
];
if (!remoteInboundRTPStreamsStats.length) {
return issues;
}
const previousStats = this.getLastProcessedStats(data.connection.id);
if (!previousStats) {
return issues;
}
const previousRemoteInboundRTPStreamsStats = [
...previousStats.remote?.audio.inbound || [],
...previousStats.remote?.video.inbound || [],
];
const { packetsSent } = data.connection;
const lastPacketsSent = previousStats.connection.packetsSent;
const rtpNetworkStats = remoteInboundRTPStreamsStats.reduce((stats, currentStreamStats) => {
const previousStreamStats = previousRemoteInboundRTPStreamsStats
.find((stream) => stream.ssrc === currentStreamStats.ssrc);
return {
sumJitter: stats.sumJitter + currentStreamStats.jitter,
packetsLost: stats.packetsLost + currentStreamStats.packetsLost,
lastPacketsLost: stats.lastPacketsLost + (previousStreamStats?.packetsLost || 0),
};
}, {
sumJitter: 0,
packetsLost: 0,
lastPacketsLost: 0,
});
const rtt = (1e3 * data.connection.currentRoundTripTime) || 0;
const { sumJitter } = rtpNetworkStats;
const avgJitter = sumJitter / remoteInboundRTPStreamsStats.length;
const deltaPacketSent = packetsSent - lastPacketsSent;
const deltaPacketLost = rtpNetworkStats.packetsLost - rtpNetworkStats.lastPacketsLost;
const packetLossPct = deltaPacketSent && deltaPacketLost
? Math.round((deltaPacketLost * 100) / (deltaPacketSent + deltaPacketLost))
: 0;
const isHighPacketsLoss = packetLossPct > this.#highPacketLossThresholdPct;
const isHighJitter = avgJitter >= this.#highJitterThreshold;
const isNetworkMediaLatencyIssue = isHighPacketsLoss && isHighJitter;
const isNetworkIssue = (!isHighPacketsLoss && isHighJitter) || isHighJitter || isHighPacketsLoss;
const statsSample = {
rtt,
avgJitter,
packetLossPct,
};
if (isNetworkMediaLatencyIssue) {
issues.push({
statsSample,
type: IssueType.Network,
reason: IssueReason.OutboundNetworkMediaLatency,
iceCandidate: data.connection.local.id,
});
}
if (isNetworkIssue) {
issues.push({
statsSample,
type: IssueType.Network,
reason: IssueReason.OutboundNetworkQuality,
iceCandidate: data.connection.local.id,
});
}
return issues;
}
}
export default OutboundNetworkIssueDetector;