hls: fix freeze in case of muxing errors (#3135) (#3150)

This commit is contained in:
Alessandro Ros 2024-03-19 14:01:14 +01:00 committed by GitHub
parent c7dbb951f3
commit 1d4ea2cd9a
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
7 changed files with 34 additions and 12 deletions

View file

@ -50,6 +50,7 @@ func (w *Writer) Error() chan error {
func (w *Writer) run() {
w.err <- w.runInner()
close(w.err)
}
func (w *Writer) runInner() error {

View file

@ -0,0 +1,22 @@
package asyncwriter
import (
"fmt"
"testing"
"github.com/stretchr/testify/require"
)
func TestAsyncWriter(t *testing.T) {
w := New(512, nil)
w.Start()
defer w.Stop()
w.Push(func() error {
return fmt.Errorf("testerror")
})
err := <-w.Error()
require.EqualError(t, err, "testerror")
}

View file

@ -165,6 +165,12 @@ func (m *muxer) runInner() error {
recreateTimer = emptyTimer()
}
defer func() {
if mi != nil {
mi.close()
}
}()
var activityCheckTimer *time.Timer
if m.remoteAddr != "" {
activityCheckTimer = time.NewTimer(closeCheckPeriod)
@ -178,13 +184,12 @@ func (m *muxer) runInner() error {
req.res <- mi
case err := <-instanceError:
mi.close()
if m.remoteAddr != "" {
return err
}
m.Log(logger.Error, err.Error())
mi.close()
mi = nil
instanceError = make(chan error)
recreateTimer = time.NewTimer(recreatePause)
@ -215,17 +220,11 @@ func (m *muxer) runInner() error {
case <-activityCheckTimer.C:
t := time.Unix(0, atomic.LoadInt64(m.lastRequestTime))
if time.Since(t) >= closeAfterInactivity {
if mi != nil {
mi.close()
}
return fmt.Errorf("not used anymore")
}
activityCheckTimer = time.NewTimer(closeCheckPeriod)
case <-m.ctx.Done():
if mi != nil {
mi.close()
}
return errors.New("terminated")
}
}

View file

@ -237,10 +237,10 @@ func (c *conn) runRead(conn *rtmp.Conn, u *url.URL) error {
c.nconn.SetReadDeadline(time.Time{})
writer.Start()
defer writer.Stop()
select {
case <-c.ctx.Done():
writer.Stop()
return fmt.Errorf("terminated")
case err := <-writer.Error():

View file

@ -327,10 +327,10 @@ func (c *conn) runRead(req srtNewConnReq, streamID *streamID) (bool, error) {
sconn.SetReadDeadline(time.Time{})
writer.Start()
defer writer.Stop()
select {
case <-c.ctx.Done():
writer.Stop()
return true, fmt.Errorf("terminated")
case err := <-writer.Error():

View file

@ -612,17 +612,16 @@ func (s *session) runRead() (int, error) {
defer onUnreadHook()
writer.Start()
defer writer.Stop()
select {
case <-pc.Disconnected():
writer.Stop()
return 0, fmt.Errorf("peer connection closed")
case err := <-writer.Error():
return 0, err
case <-s.ctx.Done():
writer.Stop()
return 0, fmt.Errorf("terminated")
}
}

View file

@ -69,6 +69,7 @@ func (t *SourceTester) SetReady(req defs.PathSourceStaticSetReadyReq) defs.PathS
)
t.writer = asyncwriter.New(2048, t)
t.stream.AddReader(t.writer, req.Desc.Medias[0], req.Desc.Medias[0].Formats[0], func(u unit.Unit) error {
t.Unit <- u
close(t.Unit)