Files
monibuca/plugin/mp4/pkg/record.go
2024-08-25 11:33:10 +08:00

95 lines
2.6 KiB
Go

package mp4
import (
"os"
"time"
"m7s.live/m7s/v5"
"m7s.live/m7s/v5/pkg"
"m7s.live/m7s/v5/pkg/codec"
"m7s.live/m7s/v5/pkg/task"
"m7s.live/m7s/v5/plugin/mp4/pkg/box"
)
type WriteTrailerQueueTask struct {
task.Work
}
var writeTrailerQueueTask WriteTrailerQueueTask
func init() {
m7s.Servers.AddTaskLazy(&writeTrailerQueueTask)
}
func NewRecorder() *Recorder {
return &Recorder{}
}
type Recorder struct {
m7s.DefaultRecorder
}
type writeTrailerTask struct {
task.Task
muxer *box.Movmuxer
file *os.File
}
func (task *writeTrailerTask) Start() (err error) {
err = task.muxer.WriteTrailer()
if err != nil {
task.Error("write trailer", "err", err)
} else {
task.Info("write trailer")
}
return task.file.Close()
}
func (r *Recorder) Run() (err error) {
ctx := &r.RecordJob
var file *os.File
var muxer *box.Movmuxer
var audioId, videoId uint32
// TODO: fragment
if file, err = os.OpenFile(ctx.FilePath, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0666); err != nil {
return
}
muxer, err = box.CreateMp4Muxer(file)
task := &writeTrailerTask{
file: file,
muxer: muxer,
}
task.Logger = r.Logger
defer writeTrailerQueueTask.AddTask(task)
ar, vr := ctx.Subscriber.AudioReader, ctx.Subscriber.VideoReader
if ar != nil {
audioTrack := ar.Track
switch ctx := audioTrack.ICodecCtx.GetBase().(type) {
case *codec.AACCtx:
audioId = muxer.AddAudioTrack(box.MP4_CODEC_AAC, box.WithExtraData(ctx.ConfigBytes))
case *codec.PCMACtx:
audioId = muxer.AddAudioTrack(box.MP4_CODEC_G711A, box.WithAudioSampleRate(uint32(ctx.SampleRate)), box.WithAudioChannelCount(uint8(ctx.Channels)), box.WithAudioSampleBits(uint8(ctx.SampleSize)))
case *codec.PCMUCtx:
audioId = muxer.AddAudioTrack(box.MP4_CODEC_G711U, box.WithAudioSampleRate(uint32(ctx.SampleRate)), box.WithAudioChannelCount(uint8(ctx.Channels)), box.WithAudioSampleBits(uint8(ctx.SampleSize)))
}
}
if vr != nil {
videoTrack := vr.Track
switch ctx := videoTrack.ICodecCtx.GetBase().(type) {
case *codec.H264Ctx:
videoId = muxer.AddVideoTrack(box.MP4_CODEC_H264, box.WithExtraData(ctx.Record))
case *codec.H265Ctx:
videoId = muxer.AddVideoTrack(box.MP4_CODEC_H265, box.WithExtraData(ctx.Record))
}
}
return m7s.PlayBlock(ctx.Subscriber, func(audio *pkg.RawAudio) error {
return muxer.WriteAudio(audioId, audio.ToBytes(), uint64(audio.Timestamp/time.Millisecond))
}, func(video *pkg.H26xFrame) error {
var nalus [][]byte
for _, nalu := range video.Nalus {
nalus = append(nalus, nalu.ToBytes())
}
return muxer.WriteVideo(videoId, nalus, uint64(video.Timestamp/time.Millisecond), uint64(video.CTS/time.Millisecond))
})
}