2022-10-17 13:39:31 +02:00
|
|
|
package webrtc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"io"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/pion/rtcp"
|
|
|
|
"github.com/pion/webrtc/v3"
|
|
|
|
"github.com/pion/webrtc/v3/pkg/media"
|
|
|
|
"github.com/rs/zerolog"
|
|
|
|
|
|
|
|
"github.com/demodesk/neko/pkg/types"
|
|
|
|
"github.com/demodesk/neko/pkg/types/codec"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Track struct {
|
2023-03-07 00:08:53 +01:00
|
|
|
logger zerolog.Logger
|
|
|
|
track *webrtc.TrackLocalStaticSample
|
|
|
|
|
|
|
|
rtcpCh chan []rtcp.Packet
|
|
|
|
sample chan types.Sample
|
2023-02-20 19:48:04 +01:00
|
|
|
|
|
|
|
paused bool
|
2022-10-17 13:39:31 +02:00
|
|
|
stream types.StreamSinkManager
|
|
|
|
streamMu sync.Mutex
|
|
|
|
}
|
|
|
|
|
2023-04-17 01:21:20 +02:00
|
|
|
type trackOption func(*Track)
|
2023-02-06 19:45:51 +01:00
|
|
|
|
2023-04-17 01:21:20 +02:00
|
|
|
func WithRtcpChan(rtcp chan []rtcp.Packet) trackOption {
|
2023-03-07 00:08:53 +01:00
|
|
|
return func(t *Track) {
|
|
|
|
t.rtcpCh = rtcp
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-17 01:21:20 +02:00
|
|
|
func NewTrack(logger zerolog.Logger, codec codec.RTPCodec, connection *webrtc.PeerConnection, opts ...trackOption) (*Track, error) {
|
2022-10-17 13:39:31 +02:00
|
|
|
id := codec.Type.String()
|
|
|
|
track, err := webrtc.NewTrackLocalStaticSample(codec.Capability, id, "stream")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
t := &Track{
|
2023-04-17 01:21:20 +02:00
|
|
|
logger: logger.With().Str("id", id).Logger(),
|
2022-10-17 13:39:31 +02:00
|
|
|
track: track,
|
2023-04-17 01:21:20 +02:00
|
|
|
rtcpCh: nil,
|
2023-03-07 00:08:53 +01:00
|
|
|
sample: make(chan types.Sample),
|
2022-10-17 13:39:31 +02:00
|
|
|
}
|
|
|
|
|
2023-02-06 19:45:51 +01:00
|
|
|
for _, opt := range opts {
|
|
|
|
opt(t)
|
|
|
|
}
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
sender, err := connection.AddTrack(t.track)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
go t.rtcpReader(sender)
|
2023-03-07 00:08:53 +01:00
|
|
|
go t.sampleReader()
|
2022-10-17 13:39:31 +02:00
|
|
|
|
|
|
|
return t, nil
|
|
|
|
}
|
|
|
|
|
2023-03-07 00:08:53 +01:00
|
|
|
func (t *Track) Shutdown() {
|
|
|
|
t.RemoveStream()
|
|
|
|
close(t.sample)
|
|
|
|
}
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
func (t *Track) rtcpReader(sender *webrtc.RTPSender) {
|
|
|
|
for {
|
2023-02-14 21:19:14 +01:00
|
|
|
packets, _, err := sender.ReadRTCP()
|
2022-10-17 13:39:31 +02:00
|
|
|
if err != nil {
|
2023-03-07 00:08:53 +01:00
|
|
|
if errors.Is(err, io.EOF) || errors.Is(err, io.ErrClosedPipe) {
|
|
|
|
t.logger.Debug().Msg("track rtcp reader closed")
|
2022-10-17 13:39:31 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-03-07 00:08:53 +01:00
|
|
|
t.logger.Warn().Err(err).Msg("failed to read track rtcp")
|
2022-10-17 13:39:31 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-03-07 00:08:53 +01:00
|
|
|
if t.rtcpCh != nil {
|
|
|
|
t.rtcpCh <- packets
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
// --- sample ---
|
|
|
|
|
2023-03-07 00:08:53 +01:00
|
|
|
func (t *Track) sampleReader() {
|
|
|
|
for {
|
|
|
|
sample, ok := <-t.sample
|
|
|
|
if !ok {
|
|
|
|
t.logger.Debug().Msg("track sample reader closed")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err := t.track.WriteSample(media.Sample{
|
|
|
|
Data: sample.Data,
|
|
|
|
Duration: sample.Duration,
|
|
|
|
Timestamp: sample.Timestamp,
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil && !errors.Is(err, io.ErrClosedPipe) {
|
|
|
|
t.logger.Warn().Err(err).Msg("failed to write sample to track")
|
2022-10-17 13:39:31 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
func (t *Track) WriteSample(sample types.Sample) {
|
|
|
|
t.sample <- sample
|
|
|
|
}
|
|
|
|
|
|
|
|
// --- stream ---
|
|
|
|
|
2023-02-06 19:45:51 +01:00
|
|
|
func (t *Track) SetStream(stream types.StreamSinkManager) (bool, error) {
|
2022-10-17 13:39:31 +02:00
|
|
|
t.streamMu.Lock()
|
|
|
|
defer t.streamMu.Unlock()
|
|
|
|
|
2022-10-23 11:31:25 +02:00
|
|
|
// if we already listen to the stream, do nothing
|
|
|
|
if t.stream == stream {
|
2023-02-06 19:45:51 +01:00
|
|
|
return false, nil
|
2022-10-23 11:31:25 +02:00
|
|
|
}
|
|
|
|
|
2023-02-20 19:48:04 +01:00
|
|
|
// if paused, we switch the stream but don't add the listener
|
|
|
|
if t.paused {
|
|
|
|
t.stream = stream
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
var err error
|
|
|
|
if t.stream != nil {
|
2023-03-07 00:08:53 +01:00
|
|
|
err = t.stream.MoveListenerTo(t, stream)
|
2022-10-17 13:39:31 +02:00
|
|
|
} else {
|
2023-03-07 00:08:53 +01:00
|
|
|
err = stream.AddListener(t)
|
2022-10-17 13:39:31 +02:00
|
|
|
}
|
2023-02-06 19:45:51 +01:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
2022-10-17 13:39:31 +02:00
|
|
|
}
|
|
|
|
|
2023-02-06 19:45:51 +01:00
|
|
|
t.stream = stream
|
|
|
|
return true, nil
|
2022-10-17 13:39:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (t *Track) RemoveStream() {
|
|
|
|
t.streamMu.Lock()
|
|
|
|
defer t.streamMu.Unlock()
|
|
|
|
|
2023-04-17 01:21:20 +02:00
|
|
|
// if there is no stream, or paused we don't need to remove the listener
|
2023-02-20 19:48:04 +01:00
|
|
|
if t.stream == nil || t.paused {
|
2023-04-17 01:21:20 +02:00
|
|
|
t.stream = nil
|
2023-02-20 19:48:04 +01:00
|
|
|
return
|
2022-10-17 13:39:31 +02:00
|
|
|
}
|
2023-02-20 19:48:04 +01:00
|
|
|
|
2023-03-07 00:08:53 +01:00
|
|
|
err := t.stream.RemoveListener(t)
|
2023-02-20 19:48:04 +01:00
|
|
|
if err != nil {
|
|
|
|
t.logger.Warn().Err(err).Msg("failed to remove listener from stream")
|
|
|
|
}
|
|
|
|
|
|
|
|
t.stream = nil
|
2022-10-17 13:39:31 +02:00
|
|
|
}
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
func (t *Track) Stream() (types.StreamSinkManager, bool) {
|
|
|
|
t.streamMu.Lock()
|
|
|
|
defer t.streamMu.Unlock()
|
|
|
|
|
|
|
|
return t.stream, t.stream != nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// --- paused ---
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
func (t *Track) SetPaused(paused bool) {
|
2023-02-20 19:48:04 +01:00
|
|
|
t.streamMu.Lock()
|
|
|
|
defer t.streamMu.Unlock()
|
|
|
|
|
|
|
|
// if there is no state change or no stream, do nothing
|
|
|
|
if t.paused == paused || t.stream == nil {
|
2023-06-26 21:27:14 +02:00
|
|
|
t.paused = paused
|
2023-02-20 19:48:04 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var err error
|
|
|
|
if paused {
|
2023-03-07 00:08:53 +01:00
|
|
|
err = t.stream.RemoveListener(t)
|
2023-02-20 19:48:04 +01:00
|
|
|
} else {
|
2023-03-07 00:08:53 +01:00
|
|
|
err = t.stream.AddListener(t)
|
2023-02-20 19:48:04 +01:00
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
t.logger.Warn().Err(err).Msg("failed to change listener state")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-10-17 13:39:31 +02:00
|
|
|
t.paused = paused
|
|
|
|
}
|
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
func (t *Track) Paused() bool {
|
|
|
|
t.streamMu.Lock()
|
|
|
|
defer t.streamMu.Unlock()
|
2023-02-06 19:45:51 +01:00
|
|
|
|
2023-05-15 19:29:39 +02:00
|
|
|
return t.paused
|
2023-02-06 19:45:51 +01:00
|
|
|
}
|