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 *sample, 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) }