2021-09-05 15:35:04 +00:00
|
|
|
package core
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/aler9/gortsplib"
|
|
|
|
|
|
|
|
"github.com/aler9/rtsp-simple-server/internal/hls"
|
|
|
|
"github.com/aler9/rtsp-simple-server/internal/logger"
|
|
|
|
"github.com/aler9/rtsp-simple-server/internal/rtcpsenderset"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
hlsSourceRetryPause = 5 * time.Second
|
|
|
|
)
|
|
|
|
|
|
|
|
type hlsSourceParent interface {
|
2021-10-27 19:01:00 +00:00
|
|
|
log(logger.Level, string, ...interface{})
|
|
|
|
onSourceStaticSetReady(req pathSourceStaticSetReadyReq) pathSourceStaticSetReadyRes
|
2021-09-05 15:35:04 +00:00
|
|
|
OnSourceStaticSetNotReady(req pathSourceStaticSetNotReadyReq)
|
|
|
|
}
|
|
|
|
|
|
|
|
type hlsSource struct {
|
2021-10-25 19:13:02 +00:00
|
|
|
ur string
|
|
|
|
fingerprint string
|
|
|
|
wg *sync.WaitGroup
|
|
|
|
parent hlsSourceParent
|
2021-09-05 15:35:04 +00:00
|
|
|
|
|
|
|
ctx context.Context
|
|
|
|
ctxCancel func()
|
|
|
|
}
|
|
|
|
|
|
|
|
func newHLSSource(
|
|
|
|
parentCtx context.Context,
|
|
|
|
ur string,
|
2021-10-25 19:13:02 +00:00
|
|
|
fingerprint string,
|
2021-09-05 15:35:04 +00:00
|
|
|
wg *sync.WaitGroup,
|
|
|
|
parent hlsSourceParent) *hlsSource {
|
|
|
|
ctx, ctxCancel := context.WithCancel(parentCtx)
|
|
|
|
|
|
|
|
s := &hlsSource{
|
2021-10-25 19:13:02 +00:00
|
|
|
ur: ur,
|
|
|
|
fingerprint: fingerprint,
|
|
|
|
wg: wg,
|
|
|
|
parent: parent,
|
|
|
|
ctx: ctx,
|
|
|
|
ctxCancel: ctxCancel,
|
2021-09-05 15:35:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
s.Log(logger.Info, "started")
|
|
|
|
|
|
|
|
s.wg.Add(1)
|
|
|
|
go s.run()
|
|
|
|
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
2021-10-27 19:01:00 +00:00
|
|
|
func (s *hlsSource) close() {
|
2021-09-05 15:35:04 +00:00
|
|
|
s.Log(logger.Info, "stopped")
|
|
|
|
s.ctxCancel()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *hlsSource) Log(level logger.Level, format string, args ...interface{}) {
|
2021-10-27 19:01:00 +00:00
|
|
|
s.parent.log(level, "[hls source] "+format, args...)
|
2021-09-05 15:35:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *hlsSource) run() {
|
|
|
|
defer s.wg.Done()
|
|
|
|
|
|
|
|
outer:
|
|
|
|
for {
|
|
|
|
ok := s.runInner()
|
|
|
|
if !ok {
|
|
|
|
break outer
|
|
|
|
}
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-time.After(hlsSourceRetryPause):
|
|
|
|
case <-s.ctx.Done():
|
|
|
|
break outer
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s.ctxCancel()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *hlsSource) runInner() bool {
|
|
|
|
var stream *stream
|
|
|
|
var rtcpSenders *rtcpsenderset.RTCPSenderSet
|
|
|
|
var videoTrackID int
|
|
|
|
var audioTrackID int
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if stream != nil {
|
|
|
|
s.parent.OnSourceStaticSetNotReady(pathSourceStaticSetNotReadyReq{Source: s})
|
|
|
|
rtcpSenders.Close()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
onTracks := func(videoTrack *gortsplib.Track, audioTrack *gortsplib.Track) error {
|
|
|
|
var tracks gortsplib.Tracks
|
|
|
|
|
|
|
|
if videoTrack != nil {
|
|
|
|
videoTrackID = len(tracks)
|
|
|
|
tracks = append(tracks, videoTrack)
|
|
|
|
}
|
|
|
|
|
|
|
|
if audioTrack != nil {
|
|
|
|
audioTrackID = len(tracks)
|
|
|
|
tracks = append(tracks, audioTrack)
|
|
|
|
}
|
|
|
|
|
2021-10-27 19:01:00 +00:00
|
|
|
res := s.parent.onSourceStaticSetReady(pathSourceStaticSetReadyReq{
|
2021-09-05 15:35:04 +00:00
|
|
|
Source: s,
|
|
|
|
Tracks: tracks,
|
|
|
|
})
|
|
|
|
if res.Err != nil {
|
|
|
|
return res.Err
|
|
|
|
}
|
|
|
|
|
|
|
|
s.Log(logger.Info, "ready")
|
|
|
|
|
|
|
|
stream = res.Stream
|
|
|
|
rtcpSenders = rtcpsenderset.New(tracks, stream.onFrame)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
onFrame := func(isVideo bool, payload []byte) {
|
|
|
|
var trackID int
|
|
|
|
if isVideo {
|
|
|
|
trackID = videoTrackID
|
|
|
|
} else {
|
|
|
|
trackID = audioTrackID
|
|
|
|
}
|
|
|
|
|
|
|
|
if stream != nil {
|
|
|
|
rtcpSenders.OnFrame(trackID, gortsplib.StreamTypeRTP, payload)
|
|
|
|
stream.onFrame(trackID, gortsplib.StreamTypeRTP, payload)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
c := hls.NewClient(
|
|
|
|
s.ur,
|
2021-10-25 19:13:02 +00:00
|
|
|
s.fingerprint,
|
2021-09-05 15:35:04 +00:00
|
|
|
onTracks,
|
|
|
|
onFrame,
|
|
|
|
s,
|
|
|
|
)
|
|
|
|
|
|
|
|
select {
|
|
|
|
case err := <-c.Wait():
|
|
|
|
s.Log(logger.Info, "ERR: %v", err)
|
|
|
|
return true
|
|
|
|
|
|
|
|
case <-s.ctx.Done():
|
|
|
|
c.Close()
|
|
|
|
<-c.Wait()
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-27 19:01:00 +00:00
|
|
|
// onSourceAPIDescribe implements source.
|
|
|
|
func (*hlsSource) onSourceAPIDescribe() interface{} {
|
2021-09-05 15:35:04 +00:00
|
|
|
return struct {
|
|
|
|
Type string `json:"type"`
|
|
|
|
}{"hlsSource"}
|
|
|
|
}
|