mirror of
https://github.com/bluenviron/mediamtx.git
synced 2025-12-20 02:00:05 -08:00
the server now detects when system time changes too much and restarts recordings when that happens.
238 lines
4.5 KiB
Go
238 lines
4.5 KiB
Go
package recorder
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
amp4 "github.com/abema/go-mp4"
|
|
"github.com/google/uuid"
|
|
|
|
"github.com/bluenviron/mediacommon/v2/pkg/formats/fmp4"
|
|
"github.com/bluenviron/mediacommon/v2/pkg/formats/fmp4/seekablebuffer"
|
|
"github.com/bluenviron/mediamtx/internal/logger"
|
|
"github.com/bluenviron/mediamtx/internal/recordstore"
|
|
)
|
|
|
|
func writeInit(
|
|
f io.Writer,
|
|
streamID uuid.UUID,
|
|
segmentNumber uint64,
|
|
dts time.Duration,
|
|
ntp time.Time,
|
|
tracks []*formatFMP4Track,
|
|
) error {
|
|
fmp4Tracks := make([]*fmp4.InitTrack, len(tracks))
|
|
for i, track := range tracks {
|
|
fmp4Tracks[i] = track.initTrack
|
|
}
|
|
|
|
init := fmp4.Init{
|
|
Tracks: fmp4Tracks,
|
|
UserData: []amp4.IBox{
|
|
&recordstore.Mtxi{
|
|
FullBox: amp4.FullBox{
|
|
Version: 0,
|
|
},
|
|
StreamID: streamID,
|
|
SegmentNumber: segmentNumber,
|
|
DTS: int64(dts),
|
|
NTP: ntp.UnixNano(),
|
|
},
|
|
},
|
|
}
|
|
|
|
var buf seekablebuffer.Buffer
|
|
err := init.Marshal(&buf)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = f.Write(buf.Bytes())
|
|
return err
|
|
}
|
|
|
|
func writeDuration(f io.ReadWriteSeeker, d time.Duration) error {
|
|
_, err := f.Seek(0, io.SeekStart)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// check and skip ftyp header and content
|
|
|
|
buf := make([]byte, 8)
|
|
_, err = io.ReadFull(f, buf)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if !bytes.Equal(buf[4:], []byte{'f', 't', 'y', 'p'}) {
|
|
return fmt.Errorf("ftyp box not found")
|
|
}
|
|
|
|
ftypSize := uint32(buf[0])<<24 | uint32(buf[1])<<16 | uint32(buf[2])<<8 | uint32(buf[3])
|
|
|
|
_, err = f.Seek(int64(ftypSize), io.SeekStart)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// check and skip moov header
|
|
|
|
_, err = io.ReadFull(f, buf)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if !bytes.Equal(buf[4:], []byte{'m', 'o', 'o', 'v'}) {
|
|
return fmt.Errorf("moov box not found")
|
|
}
|
|
|
|
moovSize := uint32(buf[0])<<24 | uint32(buf[1])<<16 | uint32(buf[2])<<8 | uint32(buf[3])
|
|
|
|
moovPos, err := f.Seek(8, io.SeekCurrent)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var mvhd amp4.Mvhd
|
|
_, err = amp4.Unmarshal(f, uint64(moovSize-8), &mvhd, amp4.Context{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
mvhd.DurationV0 = uint32(d / time.Millisecond)
|
|
|
|
_, err = f.Seek(moovPos, io.SeekStart)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = amp4.Marshal(f, &mvhd, amp4.Context{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type formatFMP4Segment struct {
|
|
f *formatFMP4
|
|
startDTS time.Duration
|
|
startNTP time.Time
|
|
number uint64
|
|
|
|
path string
|
|
fi *os.File
|
|
curPart *formatFMP4Part
|
|
endDTS time.Duration
|
|
nextPartNumber uint32
|
|
}
|
|
|
|
func (s *formatFMP4Segment) initialize() {
|
|
s.endDTS = s.startDTS
|
|
}
|
|
|
|
func (s *formatFMP4Segment) close() error {
|
|
var err error
|
|
|
|
if s.curPart != nil {
|
|
err = s.closeCurPart()
|
|
}
|
|
|
|
if s.fi != nil {
|
|
s.f.ri.Log(logger.Debug, "closing segment %s", s.path)
|
|
|
|
// write overall duration in the header to speed up the playback server
|
|
duration := s.endDTS - s.startDTS
|
|
err2 := writeDuration(s.fi, duration)
|
|
if err == nil {
|
|
err = err2
|
|
}
|
|
|
|
err2 = s.fi.Close()
|
|
if err == nil {
|
|
err = err2
|
|
}
|
|
|
|
if err2 == nil {
|
|
s.f.ri.onSegmentComplete(s.path, duration)
|
|
}
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
func (s *formatFMP4Segment) closeCurPart() error {
|
|
if s.fi == nil {
|
|
s.path = recordstore.Path{Start: s.startNTP}.Encode(s.f.ri.pathFormat2)
|
|
s.f.ri.Log(logger.Debug, "creating segment %s", s.path)
|
|
|
|
err := os.MkdirAll(filepath.Dir(s.path), 0o755)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
fi, err := os.Create(s.path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
s.f.ri.onSegmentCreate(s.path)
|
|
|
|
err = writeInit(
|
|
fi,
|
|
s.f.ri.streamID,
|
|
s.number,
|
|
s.startDTS,
|
|
s.startNTP,
|
|
s.f.tracks)
|
|
if err != nil {
|
|
fi.Close()
|
|
return err
|
|
}
|
|
|
|
s.fi = fi
|
|
}
|
|
|
|
return s.curPart.close(s.fi)
|
|
}
|
|
|
|
func (s *formatFMP4Segment) write(track *formatFMP4Track, sample *formatFMP4Sample, dts time.Duration) error {
|
|
endDTS := dts + timestampToDuration(int64(sample.Duration), int(track.initTrack.TimeScale))
|
|
if endDTS > s.endDTS {
|
|
s.endDTS = endDTS
|
|
}
|
|
|
|
if s.curPart == nil {
|
|
s.curPart = &formatFMP4Part{
|
|
maxPartSize: s.f.ri.maxPartSize,
|
|
segmentStartDTS: s.startDTS,
|
|
number: s.nextPartNumber,
|
|
startDTS: dts,
|
|
}
|
|
s.curPart.initialize()
|
|
s.nextPartNumber++
|
|
} else if s.curPart.duration() >= s.f.ri.partDuration {
|
|
err := s.closeCurPart()
|
|
s.curPart = nil
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
s.curPart = &formatFMP4Part{
|
|
maxPartSize: s.f.ri.maxPartSize,
|
|
segmentStartDTS: s.startDTS,
|
|
number: s.nextPartNumber,
|
|
startDTS: dts,
|
|
}
|
|
s.curPart.initialize()
|
|
s.nextPartNumber++
|
|
}
|
|
|
|
return s.curPart.write(track, sample, dts)
|
|
}
|