neko/internal/webrtc/peer.go

324 lines
7.1 KiB
Go
Raw Normal View History

package webrtc
2021-06-30 00:04:41 +02:00
import (
"bytes"
"encoding/binary"
2021-06-30 00:04:41 +02:00
"sync"
2023-04-17 01:10:20 +02:00
"time"
2021-06-30 00:04:41 +02:00
2023-04-10 21:44:17 +02:00
"github.com/pion/interceptor/pkg/cc"
2023-04-10 21:21:11 +02:00
"github.com/pion/rtcp"
2021-06-30 00:04:41 +02:00
"github.com/pion/webrtc/v3"
2021-08-29 18:59:46 +02:00
"github.com/rs/zerolog"
2022-02-10 00:12:30 +01:00
"github.com/demodesk/neko/internal/webrtc/payload"
"github.com/demodesk/neko/pkg/types"
2023-04-10 21:44:17 +02:00
"github.com/demodesk/neko/pkg/types/event"
"github.com/demodesk/neko/pkg/types/message"
2021-06-30 00:04:41 +02:00
)
2023-04-17 01:21:32 +02:00
const (
// how often to read and process bandwidth estimation reports
estimatorReadInterval = 250 * time.Millisecond
)
2020-11-25 18:41:40 +01:00
type WebRTCPeerCtx struct {
2023-04-10 21:21:11 +02:00
mu sync.Mutex
logger zerolog.Logger
2023-04-10 21:37:39 +02:00
session types.Session
metrics *metrics
2023-04-10 21:21:11 +02:00
connection *webrtc.PeerConnection
2023-04-10 21:44:17 +02:00
estimator cc.BandwidthEstimator
2023-04-10 21:21:11 +02:00
// tracks & channels
audioTrack *Track
videoTrack *Track
dataChannel *webrtc.DataChannel
rtcpChannel chan []rtcp.Packet
// config
2023-04-17 01:10:20 +02:00
iceTrickle bool
estimatorPassive bool
}
2023-04-16 23:34:02 +02:00
//
// connection
//
func (peer *WebRTCPeerCtx) CreateOffer(ICERestart bool) (*webrtc.SessionDescription, error) {
2021-06-30 00:04:41 +02:00
peer.mu.Lock()
defer peer.mu.Unlock()
2021-06-27 22:02:05 +02:00
offer, err := peer.connection.CreateOffer(&webrtc.OfferOptions{
ICERestart: ICERestart,
})
if err != nil {
return nil, err
}
2021-11-28 18:36:47 +01:00
return peer.setLocalDescription(offer)
}
func (peer *WebRTCPeerCtx) CreateAnswer() (*webrtc.SessionDescription, error) {
peer.mu.Lock()
defer peer.mu.Unlock()
answer, err := peer.connection.CreateAnswer(nil)
if err != nil {
return nil, err
}
return peer.setLocalDescription(answer)
}
func (peer *WebRTCPeerCtx) setLocalDescription(description webrtc.SessionDescription) (*webrtc.SessionDescription, error) {
if !peer.iceTrickle {
2021-06-27 22:02:05 +02:00
// Create channel that is blocked until ICE Gathering is complete
gatherComplete := webrtc.GatheringCompletePromise(peer.connection)
2021-11-28 18:36:47 +01:00
if err := peer.connection.SetLocalDescription(description); err != nil {
2021-06-27 22:02:05 +02:00
return nil, err
}
<-gatherComplete
} else {
2021-11-28 18:36:47 +01:00
if err := peer.connection.SetLocalDescription(description); err != nil {
2021-06-27 22:02:05 +02:00
return nil, err
}
}
return peer.connection.LocalDescription(), nil
}
2023-04-16 23:34:02 +02:00
func (peer *WebRTCPeerCtx) SetRemoteDescription(desc webrtc.SessionDescription) error {
2021-11-28 18:36:47 +01:00
peer.mu.Lock()
defer peer.mu.Unlock()
2023-04-16 23:34:02 +02:00
return peer.connection.SetRemoteDescription(desc)
2021-11-25 00:07:17 +01:00
}
2023-04-16 23:34:02 +02:00
func (peer *WebRTCPeerCtx) SetCandidate(candidate webrtc.ICECandidateInit) error {
2021-11-28 18:36:47 +01:00
peer.mu.Lock()
defer peer.mu.Unlock()
2023-04-16 23:34:02 +02:00
return peer.connection.AddICECandidate(candidate)
}
2023-04-16 23:34:02 +02:00
func (peer *WebRTCPeerCtx) Destroy() {
2021-11-28 18:36:47 +01:00
peer.mu.Lock()
defer peer.mu.Unlock()
2023-04-17 22:38:03 +02:00
err := peer.connection.Close()
peer.logger.Err(err).Msg("peer connection destroyed")
2021-02-02 20:43:33 +01:00
}
2023-04-17 01:10:20 +02:00
func (peer *WebRTCPeerCtx) estimatorReader() {
// if estimator is disabled, do nothing
if peer.estimator == nil {
return
}
// use a ticker to get current client target bitrate
2023-04-17 01:21:32 +02:00
ticker := time.NewTicker(estimatorReadInterval)
2023-04-17 01:10:20 +02:00
defer ticker.Stop()
for range ticker.C {
targetBitrate := peer.estimator.GetTargetBitrate()
peer.metrics.SetReceiverEstimatedTargetBitrate(float64(targetBitrate))
if peer.connection.ConnectionState() == webrtc.PeerConnectionStateClosed {
break
}
if !peer.videoTrack.VideoAuto() {
continue
}
if !peer.estimatorPassive {
err := peer.SetVideoBitrate(targetBitrate)
if err != nil {
peer.logger.Warn().Err(err).Msg("failed to set video bitrate")
}
}
}
}
2023-04-16 23:34:02 +02:00
//
// video
//
func (peer *WebRTCPeerCtx) SetVideoBitrate(peerBitrate int) error {
2021-06-30 00:04:41 +02:00
peer.mu.Lock()
defer peer.mu.Unlock()
2023-04-10 21:44:17 +02:00
// when switching from manual to auto bitrate estimation, in case the estimator is
// idle (lastBitrate > maxBitrate), we want to go back to the previous estimated bitrate
2023-04-17 01:10:20 +02:00
if peerBitrate == 0 && peer.estimator != nil && !peer.estimatorPassive {
2023-04-10 21:44:17 +02:00
peerBitrate = peer.estimator.GetTargetBitrate()
peer.logger.Debug().
Int("peer_bitrate", peerBitrate).
Msg("evaluated bitrate")
}
changed, err := peer.videoTrack.SetBitrate(peerBitrate)
if err != nil {
return err
}
if !changed {
// TODO: return error?
return nil
}
videoID := peer.videoTrack.stream.ID()
bitrate := peer.videoTrack.stream.Bitrate()
peer.metrics.SetVideoID(videoID)
peer.logger.Debug().
Int("peer_bitrate", peerBitrate).
Int("video_bitrate", bitrate).
Str("video_id", videoID).
Msg("peer bitrate triggered video stream change")
go peer.session.Send(
event.SIGNAL_VIDEO,
message.SignalVideo{
Video: videoID,
Bitrate: bitrate,
VideoAuto: peer.videoTrack.VideoAuto(),
})
return nil
}
func (peer *WebRTCPeerCtx) SetVideoID(videoID string) error {
peer.mu.Lock()
defer peer.mu.Unlock()
2023-04-10 21:44:17 +02:00
changed, err := peer.videoTrack.SetVideoID(videoID)
if err != nil {
return err
}
if !changed {
// TODO: return error?
return nil
}
bitrate := peer.videoTrack.stream.Bitrate()
peer.logger.Debug().
Str("video_id", videoID).
Int("video_bitrate", bitrate).
Msg("peer video id triggered video stream change")
go peer.session.Send(
event.SIGNAL_VIDEO,
message.SignalVideo{
Video: videoID,
Bitrate: bitrate,
VideoAuto: peer.videoTrack.VideoAuto(),
})
return nil
}
func (peer *WebRTCPeerCtx) GetVideoID() string {
peer.mu.Lock()
defer peer.mu.Unlock()
2023-04-10 21:47:22 +02:00
// TODO: Refactor.
return peer.videoTrack.stream.ID()
}
2022-03-26 23:20:38 +01:00
func (peer *WebRTCPeerCtx) SetPaused(isPaused bool) error {
peer.mu.Lock()
defer peer.mu.Unlock()
peer.logger.Info().Bool("is_paused", isPaused).Msg("set paused")
2023-04-10 21:47:22 +02:00
peer.videoTrack.SetPaused(isPaused)
peer.audioTrack.SetPaused(isPaused)
2022-03-26 23:20:38 +01:00
return nil
}
2023-04-16 23:34:02 +02:00
func (peer *WebRTCPeerCtx) SetVideoAuto(videoAuto bool) {
2023-04-17 01:10:20 +02:00
// if estimator is enabled and is not passive, enable video auto bitrate
if peer.estimator != nil && !peer.estimatorPassive {
2023-04-16 23:34:02 +02:00
peer.videoTrack.SetVideoAuto(videoAuto)
} else {
peer.logger.Warn().Msg("estimator is disabled or in passive mode, cannot change video auto")
peer.videoTrack.SetVideoAuto(false) // ensure video auto is disabled
}
}
func (peer *WebRTCPeerCtx) VideoAuto() bool {
return peer.videoTrack.VideoAuto()
}
//
// data channel
//
func (peer *WebRTCPeerCtx) SendCursorPosition(x, y int) error {
peer.mu.Lock()
defer peer.mu.Unlock()
2023-04-17 00:42:29 +02:00
// do not send cursor position to host
if peer.session.IsHost() {
return nil
}
2023-04-10 23:12:11 +02:00
header := payload.Header{
Event: payload.OP_CURSOR_POSITION,
Length: 7,
}
data := payload.CursorPosition{
X: uint16(x),
Y: uint16(y),
}
buffer := &bytes.Buffer{}
2023-04-10 23:12:11 +02:00
if err := binary.Write(buffer, binary.BigEndian, header); err != nil {
return err
}
if err := binary.Write(buffer, binary.BigEndian, data); err != nil {
return err
}
return peer.dataChannel.Send(buffer.Bytes())
}
func (peer *WebRTCPeerCtx) SendCursorImage(cur *types.CursorImage, img []byte) error {
peer.mu.Lock()
defer peer.mu.Unlock()
2023-04-10 23:12:11 +02:00
header := payload.Header{
Event: payload.OP_CURSOR_IMAGE,
Length: uint16(11 + len(img)),
}
data := payload.CursorImage{
Width: cur.Width,
Height: cur.Height,
Xhot: cur.Xhot,
Yhot: cur.Yhot,
}
buffer := &bytes.Buffer{}
2023-04-10 23:12:11 +02:00
if err := binary.Write(buffer, binary.BigEndian, header); err != nil {
return err
}
if err := binary.Write(buffer, binary.BigEndian, data); err != nil {
return err
}
if err := binary.Write(buffer, binary.BigEndian, img); err != nil {
return err
}
return peer.dataChannel.Send(buffer.Bytes())
}