2022-06-25 20:12:42 +02:00
|
|
|
package webrtc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
2023-04-10 22:24:16 +02:00
|
|
|
"time"
|
2022-06-25 20:12:42 +02:00
|
|
|
|
2022-07-14 00:58:22 +02:00
|
|
|
"github.com/demodesk/neko/pkg/types"
|
2022-07-04 01:01:03 +02:00
|
|
|
"github.com/pion/rtcp"
|
2022-06-25 20:12:42 +02:00
|
|
|
"github.com/pion/webrtc/v3"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
|
|
|
)
|
|
|
|
|
2023-04-17 01:21:32 +02:00
|
|
|
const (
|
|
|
|
// how often to read and process webrtc connection stats
|
|
|
|
connectionStatsInterval = 5 * time.Second
|
|
|
|
)
|
2023-04-10 22:24:16 +02:00
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
type metricsManager struct {
|
2022-06-25 20:12:42 +02:00
|
|
|
mu sync.Mutex
|
|
|
|
|
2023-04-10 21:37:39 +02:00
|
|
|
sessions map[string]*metrics
|
2022-06-25 20:12:42 +02:00
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func newMetricsManager() *metricsManager {
|
|
|
|
return &metricsManager{
|
2023-04-10 21:37:39 +02:00
|
|
|
sessions: map[string]*metrics{},
|
2022-06-25 20:12:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:37:39 +02:00
|
|
|
func (m *metricsManager) getBySession(session types.Session) *metrics {
|
2022-06-25 20:12:42 +02:00
|
|
|
m.mu.Lock()
|
|
|
|
defer m.mu.Unlock()
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
sessionId := session.ID()
|
|
|
|
|
|
|
|
met, ok := m.sessions[sessionId]
|
2022-06-25 20:12:42 +02:00
|
|
|
if ok {
|
|
|
|
return met
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:37:39 +02:00
|
|
|
met = &metrics{
|
2023-04-10 21:33:53 +02:00
|
|
|
sessionId: sessionId,
|
|
|
|
|
2022-06-25 20:12:42 +02:00
|
|
|
connectionState: promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "connection_state",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
|
|
|
Help: "Connection state of session.",
|
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-06-25 20:12:42 +02:00
|
|
|
},
|
|
|
|
}),
|
2022-06-30 23:54:06 +02:00
|
|
|
connectionStateCount: promauto.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "connection_state_count",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
|
|
|
Help: "Count of connection state changes for a session.",
|
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-06-30 23:54:06 +02:00
|
|
|
},
|
|
|
|
}),
|
2022-06-25 20:12:42 +02:00
|
|
|
connectionCount: promauto.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "connection_count",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
|
|
|
Help: "Connection count of a session.",
|
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-06-25 20:12:42 +02:00
|
|
|
},
|
|
|
|
}),
|
2022-06-30 23:50:47 +02:00
|
|
|
|
2022-07-03 23:06:56 +02:00
|
|
|
iceCandidates: map[string]struct{}{},
|
|
|
|
iceCandidatesMu: &sync.Mutex{},
|
2022-12-16 13:49:51 +01:00
|
|
|
iceCandidatesUdpCount: promauto.NewCounter(prometheus.CounterOpts{
|
2022-06-30 23:50:47 +02:00
|
|
|
Name: "ice_candidates_count",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
|
|
|
Help: "Count of ICE candidates sent by a remote client.",
|
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-12-16 13:49:51 +01:00
|
|
|
"protocol": "udp",
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
iceCandidatesTcpCount: promauto.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "ice_candidates_count",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
|
|
|
Help: "Count of ICE candidates sent by a remote client.",
|
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-12-16 13:49:51 +01:00
|
|
|
"protocol": "tcp",
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
|
|
|
|
iceCandidatesUsedUdp: promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "ice_candidates_used",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
|
|
|
Help: "Used ICE candidates that are currently in use.",
|
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-12-16 13:49:51 +01:00
|
|
|
"protocol": "udp",
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
iceCandidatesUsedTcp: promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "ice_candidates_used",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
|
|
|
Help: "Used ICE candidates that are currently in use.",
|
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-12-16 13:49:51 +01:00
|
|
|
"protocol": "tcp",
|
2022-06-30 23:50:47 +02:00
|
|
|
},
|
|
|
|
}),
|
|
|
|
|
2022-07-03 23:21:25 +02:00
|
|
|
videoIds: map[string]prometheus.Gauge{},
|
|
|
|
videoIdsMu: &sync.Mutex{},
|
|
|
|
|
2022-07-04 00:38:46 +02:00
|
|
|
receiverEstimatedMaximumBitrate: promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "receiver_estimated_maximum_bitrate",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
2023-05-15 19:29:39 +02:00
|
|
|
Help: "Receiver Estimated Maximum Bitrate from RTCP.",
|
2022-07-04 00:38:46 +02:00
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-07-04 00:38:46 +02:00
|
|
|
},
|
|
|
|
}),
|
2023-03-13 17:55:52 +01:00
|
|
|
receiverEstimatedTargetBitrate: promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "receiver_estimated_target_bitrate",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
|
|
|
Help: "Receiver Estimated Target Bitrate using Google's congestion control.",
|
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2023-03-13 17:55:52 +01:00
|
|
|
},
|
|
|
|
}),
|
2022-07-04 00:38:46 +02:00
|
|
|
|
2022-07-04 01:01:03 +02:00
|
|
|
receiverReportDelay: promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "receiver_report_delay",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
2023-05-15 19:29:39 +02:00
|
|
|
Help: "Receiver Report Delay from RTCP, expressed in units of 1/65536 seconds.",
|
2022-07-04 01:01:03 +02:00
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-07-04 01:01:03 +02:00
|
|
|
},
|
|
|
|
}),
|
|
|
|
receiverReportJitter: promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "receiver_report_jitter",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
2023-05-15 19:29:39 +02:00
|
|
|
Help: "Receiver Report Jitter from RTCP.",
|
2022-07-04 01:01:03 +02:00
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-07-04 01:01:03 +02:00
|
|
|
},
|
|
|
|
}),
|
|
|
|
receiverReportTotalLost: promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "receiver_report_total_lost",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
2023-05-15 19:29:39 +02:00
|
|
|
Help: "Receiver Report Total Lost from RTCP.",
|
|
|
|
ConstLabels: map[string]string{
|
|
|
|
"session_id": sessionId,
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
|
|
|
|
transportLayerNacks: promauto.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "transport_layer_nacks",
|
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
|
|
|
Help: "Transport Layer NACKs from RTCP.",
|
2022-07-04 01:01:03 +02:00
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-07-04 01:01:03 +02:00
|
|
|
},
|
|
|
|
}),
|
|
|
|
|
2022-07-01 00:16:39 +02:00
|
|
|
iceBytesSent: promauto.NewGauge(prometheus.GaugeOpts{
|
2022-07-04 00:48:29 +02:00
|
|
|
Name: "bytes_sent",
|
2022-06-25 20:12:42 +02:00
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
2022-07-04 00:48:29 +02:00
|
|
|
Help: "Sent bytes to a session.",
|
2022-06-25 20:12:42 +02:00
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-07-04 00:48:29 +02:00
|
|
|
"transport": "ice",
|
2022-06-25 20:12:42 +02:00
|
|
|
},
|
|
|
|
}),
|
2022-07-01 00:16:39 +02:00
|
|
|
iceBytesReceived: promauto.NewGauge(prometheus.GaugeOpts{
|
2022-07-04 00:48:29 +02:00
|
|
|
Name: "bytes_received",
|
2022-06-25 20:12:42 +02:00
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
2022-07-04 00:48:29 +02:00
|
|
|
Help: "Received bytes from a session.",
|
2022-07-01 00:16:39 +02:00
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-07-04 00:48:29 +02:00
|
|
|
"transport": "ice",
|
2022-07-01 00:16:39 +02:00
|
|
|
},
|
|
|
|
}),
|
|
|
|
|
|
|
|
sctpBytesSent: promauto.NewGauge(prometheus.GaugeOpts{
|
2022-07-04 00:48:29 +02:00
|
|
|
Name: "bytes_sent",
|
2022-07-01 00:16:39 +02:00
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
2022-07-04 00:48:29 +02:00
|
|
|
Help: "Sent bytes to a session.",
|
2022-07-01 00:16:39 +02:00
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-07-04 00:48:29 +02:00
|
|
|
"transport": "sctp",
|
2022-07-01 00:16:39 +02:00
|
|
|
},
|
|
|
|
}),
|
|
|
|
sctpBytesReceived: promauto.NewGauge(prometheus.GaugeOpts{
|
2022-07-04 00:48:29 +02:00
|
|
|
Name: "bytes_received",
|
2022-07-01 00:16:39 +02:00
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
2022-07-04 00:48:29 +02:00
|
|
|
Help: "Received bytes from a session.",
|
2022-06-25 20:12:42 +02:00
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": sessionId,
|
2022-07-04 00:48:29 +02:00
|
|
|
"transport": "sctp",
|
2022-06-25 20:12:42 +02:00
|
|
|
},
|
|
|
|
}),
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
m.sessions[sessionId] = met
|
2022-06-25 20:12:42 +02:00
|
|
|
return met
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
type metrics struct {
|
|
|
|
sessionId string
|
|
|
|
|
|
|
|
connectionState prometheus.Gauge
|
|
|
|
connectionStateCount prometheus.Counter
|
|
|
|
connectionCount prometheus.Counter
|
|
|
|
|
|
|
|
iceCandidates map[string]struct{}
|
|
|
|
iceCandidatesMu *sync.Mutex
|
|
|
|
iceCandidatesUdpCount prometheus.Counter
|
|
|
|
iceCandidatesTcpCount prometheus.Counter
|
|
|
|
|
|
|
|
iceCandidatesUsedUdp prometheus.Gauge
|
|
|
|
iceCandidatesUsedTcp prometheus.Gauge
|
|
|
|
|
|
|
|
videoIds map[string]prometheus.Gauge
|
|
|
|
videoIdsMu *sync.Mutex
|
|
|
|
|
|
|
|
receiverEstimatedMaximumBitrate prometheus.Gauge
|
|
|
|
receiverEstimatedTargetBitrate prometheus.Gauge
|
|
|
|
|
|
|
|
receiverReportDelay prometheus.Gauge
|
|
|
|
receiverReportJitter prometheus.Gauge
|
|
|
|
receiverReportTotalLost prometheus.Gauge
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
transportLayerNacks prometheus.Counter
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
iceBytesSent prometheus.Gauge
|
|
|
|
iceBytesReceived prometheus.Gauge
|
|
|
|
sctpBytesSent prometheus.Gauge
|
|
|
|
sctpBytesReceived prometheus.Gauge
|
|
|
|
}
|
|
|
|
|
|
|
|
func (met *metrics) reset() {
|
2022-07-04 17:47:04 +02:00
|
|
|
met.videoIdsMu.Lock()
|
|
|
|
for _, entry := range met.videoIds {
|
|
|
|
entry.Set(0)
|
|
|
|
}
|
|
|
|
met.videoIdsMu.Unlock()
|
|
|
|
|
2022-12-16 13:49:51 +01:00
|
|
|
met.iceCandidatesUsedUdp.Set(float64(0))
|
|
|
|
met.iceCandidatesUsedTcp.Set(float64(0))
|
|
|
|
|
2022-07-04 17:47:04 +02:00
|
|
|
met.receiverEstimatedMaximumBitrate.Set(0)
|
|
|
|
|
|
|
|
met.receiverReportDelay.Set(0)
|
|
|
|
met.receiverReportJitter.Set(0)
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) NewConnection() {
|
2022-06-25 20:12:42 +02:00
|
|
|
met.connectionCount.Add(1)
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) NewICECandidate(candidate webrtc.ICECandidateStats) {
|
2022-07-03 23:06:56 +02:00
|
|
|
met.iceCandidatesMu.Lock()
|
|
|
|
defer met.iceCandidatesMu.Unlock()
|
|
|
|
|
2022-12-16 13:49:51 +01:00
|
|
|
if _, found := met.iceCandidates[candidate.ID]; found {
|
2022-06-30 23:50:47 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-12-16 13:49:51 +01:00
|
|
|
met.iceCandidates[candidate.ID] = struct{}{}
|
|
|
|
if candidate.Protocol == "udp" {
|
|
|
|
met.iceCandidatesUdpCount.Add(1)
|
|
|
|
} else if candidate.Protocol == "tcp" {
|
|
|
|
met.iceCandidatesTcpCount.Add(1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) SetICECandidatesUsed(candidates []webrtc.ICECandidateStats) {
|
2022-12-16 13:49:51 +01:00
|
|
|
udp, tcp := 0, 0
|
|
|
|
for _, candidate := range candidates {
|
|
|
|
if candidate.Protocol == "udp" {
|
|
|
|
udp++
|
|
|
|
} else if candidate.Protocol == "tcp" {
|
|
|
|
tcp++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
met.iceCandidatesUsedUdp.Set(float64(udp))
|
|
|
|
met.iceCandidatesUsedTcp.Set(float64(tcp))
|
2022-06-30 23:50:47 +02:00
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) SetState(state webrtc.PeerConnectionState) {
|
2022-06-25 20:12:42 +02:00
|
|
|
switch state {
|
|
|
|
case webrtc.PeerConnectionStateNew:
|
|
|
|
met.connectionState.Set(0)
|
|
|
|
case webrtc.PeerConnectionStateConnecting:
|
|
|
|
met.connectionState.Set(4)
|
|
|
|
case webrtc.PeerConnectionStateConnected:
|
|
|
|
met.connectionState.Set(5)
|
|
|
|
case webrtc.PeerConnectionStateDisconnected:
|
|
|
|
met.connectionState.Set(3)
|
|
|
|
case webrtc.PeerConnectionStateFailed:
|
|
|
|
met.connectionState.Set(2)
|
|
|
|
case webrtc.PeerConnectionStateClosed:
|
|
|
|
met.connectionState.Set(1)
|
2023-04-10 21:33:53 +02:00
|
|
|
met.reset()
|
2022-06-25 20:12:42 +02:00
|
|
|
default:
|
|
|
|
met.connectionState.Set(-1)
|
|
|
|
}
|
2022-06-30 23:54:06 +02:00
|
|
|
|
|
|
|
met.connectionStateCount.Add(1)
|
2022-06-25 20:12:42 +02:00
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) SetVideoID(videoId string) {
|
2022-07-03 23:21:25 +02:00
|
|
|
met.videoIdsMu.Lock()
|
|
|
|
defer met.videoIdsMu.Unlock()
|
|
|
|
|
|
|
|
if _, found := met.videoIds[videoId]; !found {
|
|
|
|
met.videoIds[videoId] = promauto.NewGauge(prometheus.GaugeOpts{
|
2022-07-04 18:23:35 +02:00
|
|
|
Name: "video_listeners",
|
2022-07-03 23:21:25 +02:00
|
|
|
Namespace: "neko",
|
|
|
|
Subsystem: "webrtc",
|
2022-07-04 18:23:35 +02:00
|
|
|
Help: "Listeners for Video pipelines by a session.",
|
2022-07-03 23:21:25 +02:00
|
|
|
ConstLabels: map[string]string{
|
2023-04-10 21:33:53 +02:00
|
|
|
"session_id": met.sessionId,
|
2022-07-03 23:21:25 +02:00
|
|
|
"video_id": videoId,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
for id, entry := range met.videoIds {
|
|
|
|
if id == videoId {
|
|
|
|
entry.Set(1)
|
|
|
|
} else {
|
|
|
|
entry.Set(0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) SetReceiverEstimatedMaximumBitrate(bitrate float32) {
|
2023-03-13 17:55:52 +01:00
|
|
|
met.receiverEstimatedMaximumBitrate.Set(float64(bitrate))
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) SetReceiverEstimatedTargetBitrate(bitrate float64) {
|
2023-03-13 17:55:52 +01:00
|
|
|
met.receiverEstimatedTargetBitrate.Set(bitrate)
|
2022-07-04 00:38:46 +02:00
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) SetReceiverReport(report rtcp.ReceptionReport) {
|
2022-07-04 01:01:03 +02:00
|
|
|
met.receiverReportDelay.Set(float64(report.Delay))
|
|
|
|
met.receiverReportJitter.Set(float64(report.Jitter))
|
|
|
|
met.receiverReportTotalLost.Set(float64(report.TotalLost))
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) SetIceTransportStats(data webrtc.TransportStats) {
|
2022-07-01 00:16:39 +02:00
|
|
|
met.iceBytesSent.Set(float64(data.BytesSent))
|
|
|
|
met.iceBytesReceived.Set(float64(data.BytesReceived))
|
|
|
|
}
|
|
|
|
|
2023-04-10 21:33:53 +02:00
|
|
|
func (met *metrics) SetSctpTransportStats(data webrtc.TransportStats) {
|
2022-07-01 00:16:39 +02:00
|
|
|
met.sctpBytesSent.Set(float64(data.BytesSent))
|
|
|
|
met.sctpBytesReceived.Set(float64(data.BytesReceived))
|
2022-06-25 20:12:42 +02:00
|
|
|
}
|
2023-04-10 22:24:16 +02:00
|
|
|
|
|
|
|
//
|
|
|
|
// collectors
|
|
|
|
//
|
|
|
|
|
|
|
|
func (met *metrics) rtcpReceiver(rtcpCh chan []rtcp.Packet) {
|
|
|
|
for {
|
|
|
|
packets, ok := <-rtcpCh
|
|
|
|
if !ok {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, p := range packets {
|
|
|
|
switch rtcpPacket := p.(type) {
|
|
|
|
case *rtcp.ReceiverEstimatedMaximumBitrate: // TODO: Deprecated.
|
|
|
|
met.SetReceiverEstimatedMaximumBitrate(rtcpPacket.Bitrate)
|
|
|
|
|
|
|
|
case *rtcp.ReceiverReport:
|
|
|
|
l := len(rtcpPacket.Reports)
|
|
|
|
if l > 0 {
|
|
|
|
// use only last report
|
|
|
|
met.SetReceiverReport(rtcpPacket.Reports[l-1])
|
|
|
|
}
|
2023-05-15 19:29:39 +02:00
|
|
|
case *rtcp.TransportLayerNack:
|
|
|
|
for _, pair := range rtcpPacket.Nacks {
|
|
|
|
packetList := pair.PacketList()
|
|
|
|
met.transportLayerNacks.Add(float64(len(packetList)))
|
|
|
|
}
|
2023-04-10 22:24:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (met *metrics) connectionStats(connection *webrtc.PeerConnection) {
|
|
|
|
ticker := time.NewTicker(connectionStatsInterval)
|
|
|
|
defer ticker.Stop()
|
|
|
|
|
|
|
|
for range ticker.C {
|
|
|
|
if connection.ConnectionState() == webrtc.PeerConnectionStateClosed {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
stats := connection.GetStats()
|
|
|
|
|
|
|
|
data, ok := stats["iceTransport"].(webrtc.TransportStats)
|
|
|
|
if ok {
|
|
|
|
met.SetIceTransportStats(data)
|
|
|
|
}
|
|
|
|
|
|
|
|
data, ok = stats["sctpTransport"].(webrtc.TransportStats)
|
|
|
|
if ok {
|
|
|
|
met.SetSctpTransportStats(data)
|
|
|
|
}
|
|
|
|
|
|
|
|
remoteCandidates := map[string]webrtc.ICECandidateStats{}
|
|
|
|
nominatedRemoteCandidates := map[string]struct{}{}
|
|
|
|
for _, entry := range stats {
|
|
|
|
// only remote ice candidate stats
|
|
|
|
candidate, ok := entry.(webrtc.ICECandidateStats)
|
|
|
|
if ok && candidate.Type == webrtc.StatsTypeRemoteCandidate {
|
|
|
|
met.NewICECandidate(candidate)
|
|
|
|
remoteCandidates[candidate.ID] = candidate
|
|
|
|
}
|
|
|
|
|
|
|
|
// only nominated ice candidate pair stats
|
|
|
|
pair, ok := entry.(webrtc.ICECandidatePairStats)
|
|
|
|
if ok && pair.Nominated {
|
|
|
|
nominatedRemoteCandidates[pair.RemoteCandidateID] = struct{}{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
iceCandidatesUsed := []webrtc.ICECandidateStats{}
|
|
|
|
for id := range nominatedRemoteCandidates {
|
|
|
|
if candidate, ok := remoteCandidates[id]; ok {
|
|
|
|
iceCandidatesUsed = append(iceCandidatesUsed, candidate)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
met.SetICECandidatesUsed(iceCandidatesUsed)
|
|
|
|
}
|
|
|
|
}
|