2020-10-22 16:54:50 +02:00
|
|
|
package webrtc
|
|
|
|
|
2021-06-30 00:04:41 +02:00
|
|
|
import (
|
2022-10-17 13:39:31 +02:00
|
|
|
"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
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
"github.com/demodesk/neko/internal/config"
|
2022-10-17 13:39:31 +02:00
|
|
|
"github.com/demodesk/neko/internal/webrtc/payload"
|
2022-07-14 00:58:22 +02:00
|
|
|
"github.com/demodesk/neko/pkg/types"
|
2023-04-10 21:44:17 +02:00
|
|
|
"github.com/demodesk/neko/pkg/types/event"
|
2023-05-15 19:29:39 +02:00
|
|
|
"github.com/demodesk/neko/pkg/utils"
|
2023-04-17 01:21:32 +02:00
|
|
|
)
|
|
|
|
|
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-05-15 19:29:39 +02:00
|
|
|
// bandwidth estimator
|
|
|
|
estimator cc.BandwidthEstimator
|
|
|
|
estimateTrend *utils.TrendDetector
|
|
|
|
// stream selectors
|
2023-06-26 21:27:14 +02:00
|
|
|
video types.StreamSelectorManager
|
|
|
|
audio types.StreamSinkManager
|
2023-04-10 21:21:11 +02:00
|
|
|
// tracks & channels
|
|
|
|
audioTrack *Track
|
|
|
|
videoTrack *Track
|
|
|
|
dataChannel *webrtc.DataChannel
|
|
|
|
rtcpChannel chan []rtcp.Packet
|
|
|
|
// config
|
2023-05-15 19:29:39 +02:00
|
|
|
iceTrickle bool
|
|
|
|
estimatorConfig config.WebRTCEstimator
|
2023-06-26 21:27:14 +02:00
|
|
|
paused bool
|
2023-05-15 19:29:39 +02:00
|
|
|
videoAuto bool
|
2023-06-26 21:27:14 +02:00
|
|
|
videoDisabled bool
|
|
|
|
audioDisabled bool
|
2020-10-22 16:54:50 +02:00
|
|
|
}
|
|
|
|
|
2023-04-16 23:34:02 +02:00
|
|
|
//
|
|
|
|
// connection
|
|
|
|
//
|
|
|
|
|
2021-06-27 22:05:37 +02:00
|
|
|
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) {
|
2021-06-27 22:05:37 +02:00
|
|
|
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)
|
2020-10-22 16:54:50 +02:00
|
|
|
}
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// TODO: Add shutdown function?
|
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-05-19 17:19:38 +02:00
|
|
|
var err error
|
|
|
|
|
|
|
|
// if peer connection is not closed, close it
|
|
|
|
if peer.connection.ConnectionState() != webrtc.PeerConnectionStateClosed {
|
|
|
|
err = peer.connection.Close()
|
|
|
|
}
|
|
|
|
|
2023-04-17 22:38:03 +02:00
|
|
|
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() {
|
2023-05-15 19:29:39 +02:00
|
|
|
conf := peer.estimatorConfig
|
|
|
|
|
|
|
|
// if estimator is not in debug mode, use a nop logger
|
|
|
|
var debugLogger zerolog.Logger
|
|
|
|
if conf.Debug {
|
|
|
|
debugLogger = peer.logger.With().Str("component", "estimator").Logger().Level(zerolog.DebugLevel)
|
|
|
|
} else {
|
|
|
|
debugLogger = zerolog.Nop()
|
|
|
|
}
|
|
|
|
|
2023-04-17 01:10:20 +02:00
|
|
|
// if estimator is disabled, do nothing
|
|
|
|
if peer.estimator == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// use a ticker to get current client target bitrate
|
2023-05-15 19:29:39 +02:00
|
|
|
ticker := time.NewTicker(conf.ReadInterval)
|
2023-04-17 01:10:20 +02:00
|
|
|
defer ticker.Stop()
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// since when is the estimate stable/unstable
|
|
|
|
stableSince := time.Now() // we asume stable at start
|
|
|
|
unstableSince := time.Time{}
|
|
|
|
// since when are we neutral but cannot accomodate current bitrate
|
|
|
|
// we migt be stalled or estimator just reached zer (very bad connection)
|
|
|
|
stalledSince := time.Time{}
|
|
|
|
// when was the last upgrade/downgrade
|
|
|
|
lastUpgradeTime := time.Time{}
|
|
|
|
lastDowngradeTime := time.Time{}
|
|
|
|
|
2023-04-17 01:10:20 +02:00
|
|
|
for range ticker.C {
|
|
|
|
targetBitrate := peer.estimator.GetTargetBitrate()
|
|
|
|
peer.metrics.SetReceiverEstimatedTargetBitrate(float64(targetBitrate))
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// if peer connection is closed, stop reading
|
2023-04-17 01:10:20 +02:00
|
|
|
if peer.connection.ConnectionState() == webrtc.PeerConnectionStateClosed {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
// if estimation or video is disabled, do nothing
|
|
|
|
if !peer.videoAuto || peer.videoDisabled || peer.paused || conf.Passive {
|
2023-04-17 01:10:20 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// get trend direction to decide if we should upgrade or downgrade
|
|
|
|
peer.estimateTrend.AddValue(int64(targetBitrate))
|
|
|
|
direction := peer.estimateTrend.GetDirection()
|
|
|
|
|
|
|
|
// get current stream bitrate
|
|
|
|
stream, ok := peer.videoTrack.Stream()
|
|
|
|
if !ok {
|
|
|
|
debugLogger.Warn().Msg("looks like we don't have a stream yet, skipping bitrate estimation")
|
|
|
|
continue
|
2023-04-17 01:10:20 +02:00
|
|
|
}
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// if stream bitrate is 0, we need to wait for some time until we get a valid value
|
|
|
|
streamId, streamBitrate := stream.ID(), stream.Bitrate()
|
|
|
|
if streamBitrate == 0 {
|
|
|
|
debugLogger.Warn().Msg("looks like stream bitrate is 0, we need to wait for some time")
|
|
|
|
continue
|
|
|
|
}
|
2023-04-16 23:34:02 +02:00
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// check whats the difference between target and stream bitrate
|
|
|
|
diff := float64(targetBitrate) / float64(streamBitrate)
|
2021-06-30 00:04:41 +02:00
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
debugLogger.Info().
|
|
|
|
Float64("diff", diff).
|
|
|
|
Int("target_bitrate", targetBitrate).
|
|
|
|
Uint64("stream_bitrate", streamBitrate).
|
|
|
|
Str("direction", direction.String()).
|
|
|
|
Msg("got bitrate from estimator")
|
2023-04-10 21:44:17 +02:00
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// if we can accomodate current stream or we are not netural anymore,
|
|
|
|
// we are not stalled so we reset the stalled time
|
|
|
|
if direction != utils.TrendDirectionNeutral || diff > 1+conf.DiffThreshold {
|
|
|
|
stalledSince = time.Now()
|
|
|
|
}
|
2023-04-10 21:44:17 +02:00
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// if we are neutral and stalled for too long, we might be congesting
|
|
|
|
stalled := direction == utils.TrendDirectionNeutral && time.Since(stalledSince) > conf.StalledDuration
|
|
|
|
if stalled {
|
|
|
|
debugLogger.Warn().
|
|
|
|
Time("stalled_since", stalledSince).
|
|
|
|
Msgf("it looks like we are stalled")
|
|
|
|
}
|
2023-04-10 21:44:17 +02:00
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// if we have an downward trend or are stalled, we might be congesting
|
|
|
|
if direction == utils.TrendDirectionDownward || stalled {
|
|
|
|
// we reset the stable time because we are congesting
|
|
|
|
stableSince = time.Now()
|
|
|
|
|
|
|
|
// if we downgraded recently, we wait for some more time
|
|
|
|
if time.Since(lastDowngradeTime) < conf.DowngradeBackoff {
|
|
|
|
debugLogger.Debug().
|
|
|
|
Time("last_downgrade", lastDowngradeTime).
|
|
|
|
Msgf("downgraded recently, waiting for at least %v", conf.DowngradeBackoff)
|
|
|
|
continue
|
|
|
|
}
|
2023-04-10 21:44:17 +02:00
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// if we are not unstable but we fluctuate we should wait for some more time
|
|
|
|
if time.Since(unstableSince) < conf.UnstableDuration {
|
|
|
|
debugLogger.Debug().
|
|
|
|
Time("unstable_since", unstableSince).
|
|
|
|
Msgf("we are not unstable long enough, waiting for at least %v", conf.UnstableDuration)
|
|
|
|
continue
|
|
|
|
}
|
2023-04-10 21:44:17 +02:00
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// if we still have a big difference between target and stream bitrate, we wait for some more time
|
|
|
|
if conf.DiffThreshold >= 0 && diff > 1+conf.DiffThreshold {
|
|
|
|
debugLogger.Debug().
|
|
|
|
Float64("diff", diff).
|
|
|
|
Float64("threshold", conf.DiffThreshold).
|
|
|
|
Msgf("we still have a big difference between target and stream bitrate, " +
|
|
|
|
"therefore we still should be able to accomodate current stream")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
err := peer.SetVideo(types.PeerVideoRequest{
|
|
|
|
Selector: &types.StreamSelector{
|
|
|
|
ID: streamId,
|
|
|
|
Type: types.StreamSelectorTypeLower,
|
|
|
|
},
|
2023-05-15 19:29:39 +02:00
|
|
|
})
|
|
|
|
if err != nil && err != types.ErrWebRTCStreamNotFound {
|
|
|
|
peer.logger.Warn().Err(err).Msg("failed to downgrade video stream")
|
|
|
|
}
|
|
|
|
lastDowngradeTime = time.Now()
|
|
|
|
|
|
|
|
if err == types.ErrWebRTCStreamNotFound {
|
|
|
|
debugLogger.Info().Msg("looks like we are already on the lowest stream")
|
|
|
|
} else {
|
|
|
|
debugLogger.Info().Msg("downgraded video stream")
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// we reset the unstable time because we are not congesting
|
|
|
|
unstableSince = time.Now()
|
|
|
|
|
|
|
|
// if we have a neutral or upward trend, that means our estimate is stable
|
|
|
|
// if we are on the highest stream, we don't need to do anything
|
|
|
|
// but if there is a higher stream, we should try to upgrade and see if it works
|
|
|
|
|
|
|
|
// if we upgraded recently, we wait for some more time
|
|
|
|
if time.Since(lastUpgradeTime) < conf.UpgradeBackoff {
|
|
|
|
debugLogger.Debug().
|
|
|
|
Time("last_upgrade", lastUpgradeTime).
|
|
|
|
Msgf("upgraded recently, waiting for at least %v", conf.UpgradeBackoff)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// if we are not stable for long enough, we wait for some more time
|
|
|
|
// because bandwidth estimation might fluctuate
|
|
|
|
if time.Since(stableSince) < conf.StableDuration {
|
|
|
|
debugLogger.Debug().
|
|
|
|
Time("stable_since", stableSince).
|
|
|
|
Msgf("we are not stable long enough, waiting for at least %v", conf.StableDuration)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// upgrade only if estimated bitrate passed the threshold
|
|
|
|
if conf.DiffThreshold >= 0 && diff < 1+conf.DiffThreshold {
|
|
|
|
debugLogger.Debug().
|
|
|
|
Float64("diff", diff).
|
|
|
|
Float64("threshold", conf.DiffThreshold).
|
|
|
|
Msgf("looks like we don't have enough bitrate to accomodate higher stream, " +
|
|
|
|
"therefore we should wait for some more time")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
err := peer.SetVideo(types.PeerVideoRequest{
|
|
|
|
Selector: &types.StreamSelector{
|
|
|
|
ID: streamId,
|
|
|
|
Type: types.StreamSelectorTypeHigher,
|
|
|
|
},
|
2023-04-10 21:44:17 +02:00
|
|
|
})
|
2023-05-15 19:29:39 +02:00
|
|
|
if err != nil && err != types.ErrWebRTCStreamNotFound {
|
|
|
|
peer.logger.Warn().Err(err).Msg("failed to upgrade video stream")
|
|
|
|
}
|
|
|
|
lastUpgradeTime = time.Now()
|
2023-04-10 21:44:17 +02:00
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
if err == types.ErrWebRTCStreamNotFound {
|
|
|
|
debugLogger.Info().Msg("looks like we are already on the highest stream")
|
|
|
|
} else {
|
|
|
|
debugLogger.Info().Msg("upgraded video stream")
|
|
|
|
}
|
|
|
|
}
|
2023-02-06 19:45:51 +01:00
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
func (peer *WebRTCPeerCtx) SetPaused(isPaused bool) error {
|
|
|
|
peer.mu.Lock()
|
|
|
|
defer peer.mu.Unlock()
|
|
|
|
|
|
|
|
peer.videoTrack.SetPaused(isPaused || peer.videoDisabled)
|
|
|
|
peer.audioTrack.SetPaused(isPaused || peer.audioDisabled)
|
|
|
|
|
|
|
|
peer.logger.Info().Bool("is_paused", isPaused).Msg("set paused")
|
|
|
|
peer.paused = isPaused
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (peer *WebRTCPeerCtx) Paused() bool {
|
|
|
|
peer.mu.Lock()
|
|
|
|
defer peer.mu.Unlock()
|
|
|
|
|
|
|
|
return peer.paused
|
|
|
|
}
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
//
|
|
|
|
// video
|
|
|
|
//
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
func (peer *WebRTCPeerCtx) SetVideo(r types.PeerVideoRequest) error {
|
2023-02-06 19:45:51 +01:00
|
|
|
peer.mu.Lock()
|
|
|
|
defer peer.mu.Unlock()
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
modified := false
|
2023-05-15 19:29:39 +02:00
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
// video disabled
|
|
|
|
if r.Disabled != nil {
|
|
|
|
disabled := *r.Disabled
|
|
|
|
|
|
|
|
// update only if changed
|
|
|
|
if peer.videoDisabled != disabled {
|
|
|
|
peer.videoDisabled = disabled
|
|
|
|
peer.videoTrack.SetPaused(disabled || peer.paused)
|
|
|
|
|
|
|
|
peer.logger.Info().Bool("disabled", disabled).Msg("set video disabled")
|
|
|
|
modified = true
|
|
|
|
}
|
2023-04-10 21:44:17 +02:00
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
// video selector
|
|
|
|
if r.Selector != nil {
|
|
|
|
selector := *r.Selector
|
|
|
|
|
|
|
|
// get requested video stream from selector
|
|
|
|
stream, ok := peer.video.GetStream(selector)
|
|
|
|
if !ok {
|
|
|
|
return types.ErrWebRTCStreamNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
// set video stream to track
|
|
|
|
changed, err := peer.videoTrack.SetStream(stream)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// update only if stream changed
|
|
|
|
if changed {
|
|
|
|
videoID := stream.ID()
|
|
|
|
peer.metrics.SetVideoID(videoID)
|
|
|
|
|
|
|
|
peer.logger.Info().Str("video_id", videoID).Msg("set video")
|
|
|
|
modified = true
|
|
|
|
}
|
2023-04-10 21:44:17 +02:00
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
// video auto
|
|
|
|
if r.Auto != nil {
|
|
|
|
videoAuto := *r.Auto
|
2023-04-10 21:44:17 +02:00
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
if peer.estimator == nil || peer.estimatorConfig.Passive {
|
|
|
|
peer.logger.Warn().Msg("estimator is disabled or in passive mode, cannot change video auto")
|
|
|
|
videoAuto = false // ensure video auto is disabled
|
|
|
|
}
|
2023-04-10 21:44:17 +02:00
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
// update only if video auto changed
|
|
|
|
if peer.videoAuto != videoAuto {
|
|
|
|
peer.videoAuto = videoAuto
|
|
|
|
|
|
|
|
peer.logger.Info().Bool("video_auto", videoAuto).Msg("set video auto")
|
|
|
|
modified = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// send video signal if modified
|
|
|
|
if modified {
|
|
|
|
go func() {
|
|
|
|
// in goroutine because of mutex and we don't want to block
|
|
|
|
peer.session.Send(event.SIGNAL_VIDEO, peer.Video())
|
|
|
|
}()
|
|
|
|
}
|
2023-04-10 21:44:17 +02:00
|
|
|
|
2023-02-06 19:45:51 +01:00
|
|
|
return nil
|
2022-10-25 20:25:00 +02:00
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
func (peer *WebRTCPeerCtx) Video() types.PeerVideo {
|
2022-10-25 20:25:00 +02:00
|
|
|
peer.mu.Lock()
|
|
|
|
defer peer.mu.Unlock()
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
// get current video stream ID
|
|
|
|
ID := ""
|
2023-05-15 19:29:39 +02:00
|
|
|
stream, ok := peer.videoTrack.Stream()
|
2023-06-26 21:27:14 +02:00
|
|
|
if ok {
|
|
|
|
ID = stream.ID()
|
2023-05-15 19:29:39 +02:00
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
return types.PeerVideo{
|
|
|
|
Disabled: peer.videoDisabled,
|
|
|
|
ID: ID,
|
|
|
|
Video: ID, // TODO: Remove, used for backward compatibility
|
|
|
|
Auto: peer.videoAuto,
|
|
|
|
}
|
2021-02-04 20:39:48 +00:00
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
//
|
|
|
|
// audio
|
|
|
|
//
|
|
|
|
|
|
|
|
func (peer *WebRTCPeerCtx) SetAudio(r types.PeerAudioRequest) error {
|
2022-03-26 23:20:38 +01:00
|
|
|
peer.mu.Lock()
|
|
|
|
defer peer.mu.Unlock()
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
modified := false
|
2022-03-26 23:20:38 +01:00
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
// audio disabled
|
|
|
|
if r.Disabled != nil {
|
|
|
|
disabled := *r.Disabled
|
2023-05-15 19:29:39 +02:00
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
// update only if changed
|
|
|
|
if peer.audioDisabled != disabled {
|
|
|
|
peer.audioDisabled = disabled
|
|
|
|
peer.audioTrack.SetPaused(disabled || peer.paused)
|
2023-05-15 19:29:39 +02:00
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
peer.logger.Info().Bool("disabled", disabled).Msg("set audio disabled")
|
|
|
|
modified = true
|
|
|
|
}
|
|
|
|
}
|
2023-05-15 19:29:39 +02:00
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
// send video signal if modified
|
|
|
|
if modified {
|
|
|
|
go func() {
|
|
|
|
// in goroutine because of mutex and we don't want to block
|
|
|
|
peer.session.Send(event.SIGNAL_AUDIO, peer.Audio())
|
|
|
|
}()
|
2023-04-16 23:34:02 +02:00
|
|
|
}
|
2023-06-26 21:27:14 +02:00
|
|
|
|
|
|
|
return nil
|
2023-04-16 23:34:02 +02:00
|
|
|
}
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
func (peer *WebRTCPeerCtx) Audio() types.PeerAudio {
|
2023-05-15 19:29:39 +02:00
|
|
|
peer.mu.Lock()
|
|
|
|
defer peer.mu.Unlock()
|
|
|
|
|
2023-06-26 21:27:14 +02:00
|
|
|
return types.PeerAudio{
|
|
|
|
Disabled: peer.audioDisabled,
|
|
|
|
}
|
2023-04-16 23:34:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// data channel
|
|
|
|
//
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
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,
|
|
|
|
}
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
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)),
|
|
|
|
}
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
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())
|
|
|
|
}
|