mirror of
https://github.com/bluenviron/mediamtx
synced 2025-02-15 19:17:12 +00:00
* add hlsVariant parameter * hls: split muxer into variants * hls: implement fmp4 segments * hls muxer: implement low latency mode * hls muxer: support audio with fmp4 mode * hls muxer: rewrite file router * hls muxer: implement preload hint * hls muxer: add various error codes * hls muxer: use explicit flags * hls muxer: fix error in aac pts * hls muxer: fix sudden freezes with video+audio * hls muxer: skip empty parts * hls muxer: fix video FPS * hls muxer: add parameter hlsPartDuration * hls muxer: refactor fmp4 muxer * hls muxer: fix CAN-SKIP-UNTIL * hls muxer: refactor code * hls muxer: show only parts of last 2 segments * hls muxer: implementa playlist delta updates * hls muxer: change playlist content type * hls muxer: improve video dts precision * hls muxer: fix video sample flags * hls muxer: improve iphone audio support * hls muxer: improve mp4 timestamp precision * hls muxer: add offset between pts and dts * hls muxer: close muxer in case of error * hls muxer: stop logging requests with the info level * hls muxer: rename entry into sample * hls muxer: compensate video dts error over time * hls muxer: change default segment count * hls muxer: add starting gap * hls muxer: set default part duration to 200ms * hls muxer: fix audio-only streams on ios * hls muxer: add playsinline attribute to video tag of default web page * hls muxer: keep mpegts as the default hls variant * hls muxer: implement encryption * hls muxer: rewrite dts estimation * hls muxer: improve DTS precision * hls muxer: use right SPS/PPS for each sample * hls muxer: adjust part duration dynamically * add comments * update readme * hls muxer: fix memory leak * hls muxer: decrease ram consumption
148 lines
2.9 KiB
Go
148 lines
2.9 KiB
Go
package core
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"io"
|
|
"net"
|
|
"net/http"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/aler9/gortsplib"
|
|
"github.com/aler9/gortsplib/pkg/h264"
|
|
"github.com/asticode/go-astits"
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
type testHLSServer struct {
|
|
s *http.Server
|
|
}
|
|
|
|
func newTestHLSServer() (*testHLSServer, error) {
|
|
ln, err := net.Listen("tcp", "localhost:5780")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ts := &testHLSServer{}
|
|
|
|
gin.SetMode(gin.ReleaseMode)
|
|
router := gin.New()
|
|
router.GET("/stream.m3u8", ts.onPlaylist)
|
|
router.GET("/segment.ts", ts.onSegment)
|
|
|
|
ts.s = &http.Server{Handler: router}
|
|
go ts.s.Serve(ln)
|
|
|
|
return ts, nil
|
|
}
|
|
|
|
func (ts *testHLSServer) close() {
|
|
ts.s.Shutdown(context.Background())
|
|
}
|
|
|
|
func (ts *testHLSServer) onPlaylist(ctx *gin.Context) {
|
|
cnt := `#EXTM3U
|
|
#EXT-X-VERSION:3
|
|
#EXT-X-ALLOW-CACHE:NO
|
|
#EXT-X-TARGETDURATION:2
|
|
#EXT-X-MEDIA-SEQUENCE:0
|
|
#EXTINF:2,
|
|
segment.ts
|
|
`
|
|
|
|
ctx.Writer.Header().Set("Content-Type", `audio/mpegURL`)
|
|
io.Copy(ctx.Writer, bytes.NewReader([]byte(cnt)))
|
|
}
|
|
|
|
func (ts *testHLSServer) onSegment(ctx *gin.Context) {
|
|
ctx.Writer.Header().Set("Content-Type", `video/MP2T`)
|
|
mux := astits.NewMuxer(context.Background(), ctx.Writer)
|
|
|
|
mux.AddElementaryStream(astits.PMTElementaryStream{
|
|
ElementaryPID: 256,
|
|
StreamType: astits.StreamTypeH264Video,
|
|
})
|
|
|
|
mux.SetPCRPID(256)
|
|
|
|
mux.WriteTables()
|
|
|
|
enc, _ := h264.AnnexBEncode([][]byte{
|
|
{7, 1, 2, 3}, // SPS
|
|
{8}, // PPS
|
|
})
|
|
|
|
mux.WriteData(&astits.MuxerData{
|
|
PID: 256,
|
|
PES: &astits.PESData{
|
|
Header: &astits.PESHeader{
|
|
OptionalHeader: &astits.PESOptionalHeader{
|
|
MarkerBits: 2,
|
|
PTSDTSIndicator: astits.PTSDTSIndicatorOnlyPTS,
|
|
PTS: &astits.ClockReference{Base: int64(1 * 90000)},
|
|
},
|
|
StreamID: 224, // = video
|
|
},
|
|
Data: enc,
|
|
},
|
|
})
|
|
|
|
ctx.Writer.(http.Flusher).Flush()
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
enc, _ = h264.AnnexBEncode([][]byte{
|
|
{5}, // IDR
|
|
})
|
|
|
|
mux.WriteData(&astits.MuxerData{
|
|
PID: 256,
|
|
PES: &astits.PESData{
|
|
Header: &astits.PESHeader{
|
|
OptionalHeader: &astits.PESOptionalHeader{
|
|
MarkerBits: 2,
|
|
PTSDTSIndicator: astits.PTSDTSIndicatorOnlyPTS,
|
|
PTS: &astits.ClockReference{Base: int64(2 * 90000)},
|
|
},
|
|
StreamID: 224, // = video
|
|
},
|
|
Data: enc,
|
|
},
|
|
})
|
|
}
|
|
|
|
func TestHLSSource(t *testing.T) {
|
|
ts, err := newTestHLSServer()
|
|
require.NoError(t, err)
|
|
defer ts.close()
|
|
|
|
p, ok := newInstance("hlsDisable: yes\n" +
|
|
"rtmpDisable: yes\n" +
|
|
"paths:\n" +
|
|
" proxied:\n" +
|
|
" source: http://localhost:5780/stream.m3u8\n" +
|
|
" sourceOnDemand: yes\n")
|
|
require.Equal(t, true, ok)
|
|
defer p.close()
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
frameRecv := make(chan struct{})
|
|
|
|
c := gortsplib.Client{
|
|
OnPacketRTP: func(ctx *gortsplib.ClientOnPacketRTPCtx) {
|
|
require.Equal(t, []byte{0x05}, ctx.Packet.Payload)
|
|
close(frameRecv)
|
|
},
|
|
}
|
|
|
|
err = c.StartReading("rtsp://localhost:8554/proxied")
|
|
require.NoError(t, err)
|
|
defer c.Close()
|
|
|
|
<-frameRecv
|
|
}
|