-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathVideoDecoderIssueDetector.ts
127 lines (101 loc) · 4.02 KB
/
VideoDecoderIssueDetector.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
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
import { calculateVolatility } from '../helpers/calc';
import { isDtxLikeBehavior } from '../helpers/streams';
import {
IssueDetectorResult,
IssueReason,
IssueType,
MosQuality,
WebRTCStatsParsedWithNetworkScores,
} from '../types';
import { isSvcSpatialLayerChanged } from '../utils/video';
import BaseIssueDetector, { BaseIssueDetectorParams } from './BaseIssueDetector';
interface VideoDecoderIssueDetectorParams extends BaseIssueDetectorParams {
volatilityThreshold?: number;
affectedStreamsPercentThreshold?: number;
minMosQuality?: number;
}
const MIN_STATS_HISTORY_LENGTH = 5;
class VideoDecoderIssueDetector extends BaseIssueDetector {
readonly #volatilityThreshold: number;
readonly #affectedStreamsPercentThreshold: number;
readonly #minMosQuality: MosQuality;
constructor(params: VideoDecoderIssueDetectorParams = {}) {
super(params);
this.#volatilityThreshold = params.volatilityThreshold ?? 8;
this.#affectedStreamsPercentThreshold = params.affectedStreamsPercentThreshold ?? 30;
this.#minMosQuality = params.minMosQuality ?? MosQuality.BAD;
}
performDetection(data: WebRTCStatsParsedWithNetworkScores): IssueDetectorResult {
const allHistoricalStats = [
...this.getAllLastProcessedStats(data.connection.id),
data,
];
const isBadNetworkHappened = allHistoricalStats
.find((stat) => stat.networkScores.inbound !== undefined && stat.networkScores.inbound <= this.#minMosQuality);
if (isBadNetworkHappened) {
// do not execute detection on historical stats based on bad network quality
// to avoid false positives
return [];
}
return this.processData(data);
}
private processData(data: WebRTCStatsParsedWithNetworkScores): IssueDetectorResult {
const issues: IssueDetectorResult = [];
const allProcessedStats = [
...this.getAllLastProcessedStats(data.connection.id),
data,
];
const throtthedStreams = data.video.inbound
.map((incomeVideoStream): { ssrc: number, allFps: number[], volatility: number } | undefined => {
// At least 5 elements needed to have enough representation
if (allProcessedStats.length < MIN_STATS_HISTORY_LENGTH) {
return undefined;
}
const isSpatialLayerChanged = isSvcSpatialLayerChanged(incomeVideoStream.ssrc, allProcessedStats);
if (isSpatialLayerChanged) {
return undefined;
}
const allFps: number[] = [];
for (let i = 0; i < allProcessedStats.length - 1; i += 1) {
const videoStreamStats = allProcessedStats[i].video.inbound.find(
(stream) => stream.ssrc === incomeVideoStream.ssrc,
);
if (videoStreamStats?.framesPerSecond !== undefined) {
allFps.push(videoStreamStats.framesPerSecond);
}
}
if (allFps.length < MIN_STATS_HISTORY_LENGTH) {
return undefined;
}
const isDtx = isDtxLikeBehavior(incomeVideoStream.ssrc, allProcessedStats);
if (isDtx) {
// DTX-like behavior detected, ignoring FPS volatility check
return undefined;
}
const volatility = calculateVolatility(allFps);
if (volatility > this.#volatilityThreshold) {
return { ssrc: incomeVideoStream.ssrc, allFps, volatility };
}
return undefined;
})
.filter((throttledVideoStream) => Boolean(throttledVideoStream));
if (throtthedStreams.length === 0) {
return issues;
}
const affectedStreamsPercent = throtthedStreams.length / (data.video.inbound.length / 100);
if (affectedStreamsPercent > this.#affectedStreamsPercentThreshold) {
issues.push({
type: IssueType.CPU,
reason: IssueReason.DecoderCPUThrottling,
statsSample: {
affectedStreamsPercent,
throtthedStreams,
},
});
// clear all processed stats for this connection to avoid duplicate issues
this.deleteLastProcessedStats(data.connection.id);
}
return issues;
}
}
export default VideoDecoderIssueDetector;