m3u8d/merge.go

112 lines
2.4 KiB
Go
Raw Permalink Normal View History

2022-06-19 23:32:53 +00:00
package m3u8d
import (
"bytes"
"context"
2022-06-19 23:32:53 +00:00
"errors"
"github.com/yapingcat/gomedia/go-codec"
"github.com/yapingcat/gomedia/go-mp4"
"github.com/yapingcat/gomedia/go-mpeg2"
2022-06-19 23:32:53 +00:00
"io/ioutil"
"os"
"strconv"
)
type MergeTsFileListToSingleMp4_Req struct {
TsFileList []string
OutputMp4 string
2023-11-18 02:57:49 +00:00
Status *SpeedStatus
2022-07-23 03:21:52 +00:00
Ctx context.Context
2022-06-19 23:32:53 +00:00
}
func MergeTsFileListToSingleMp4(req MergeTsFileListToSingleMp4_Req) (err error) {
mp4file, err := os.OpenFile(req.OutputMp4, os.O_CREATE|os.O_RDWR|os.O_TRUNC, 0666)
if err != nil {
return err
}
defer mp4file.Close()
2023-11-18 02:57:49 +00:00
if req.Status != nil {
req.Status.SpeedResetBytes()
}
2022-06-21 14:17:26 +00:00
muxer, err := mp4.CreateMp4Muxer(mp4file)
if err != nil {
return err
}
2022-06-19 23:32:53 +00:00
vtid := muxer.AddVideoTrack(mp4.MP4_CODEC_H264)
atid := muxer.AddAudioTrack(mp4.MP4_CODEC_AAC)
2022-06-19 23:32:53 +00:00
demuxer := mpeg2.NewTSDemuxer()
var OnFrameErr error
2022-07-23 03:21:52 +00:00
var audioTimestamp uint64 = 0
aacSampleRate := -1
2022-06-19 23:32:53 +00:00
demuxer.OnFrame = func(cid mpeg2.TS_STREAM_TYPE, frame []byte, pts uint64, dts uint64) {
if OnFrameErr != nil {
return
}
if cid == mpeg2.TS_STREAM_AAC {
audioTimestamp = pts
2022-07-23 03:21:52 +00:00
codec.SplitAACFrame(frame, func(aac []byte) {
if aacSampleRate == -1 {
adts := codec.NewAdtsFrameHeader()
adts.Decode(aac)
aacSampleRate = codec.AACSampleIdxToSample(int(adts.Fix_Header.Sampling_frequency_index))
}
err = muxer.Write(atid, aac, audioTimestamp, audioTimestamp)
audioTimestamp += uint64(1024 * 1000 / aacSampleRate) //每帧aac采样固定为1024。aac_sampleRate 为采样率
if err != nil {
OnFrameErr = err
return
}
})
2022-06-19 23:32:53 +00:00
} else if cid == mpeg2.TS_STREAM_H264 {
2023-06-01 20:57:08 +00:00
err = muxer.Write(vtid, frame, pts, dts)
2022-07-23 03:21:52 +00:00
if err != nil {
OnFrameErr = err
return
}
2022-06-19 23:32:53 +00:00
} else {
OnFrameErr = errors.New("unknown cid " + strconv.Itoa(int(cid)))
2022-07-23 03:21:52 +00:00
return
2022-06-19 23:32:53 +00:00
}
}
for idx, tsFile := range req.TsFileList {
select {
2022-07-23 03:21:52 +00:00
case <-req.Ctx.Done():
return req.Ctx.Err()
default:
}
2022-06-19 23:32:53 +00:00
var buf []byte
buf, err = ioutil.ReadFile(tsFile)
if err != nil {
return err
}
err = demuxer.Input(bytes.NewReader(buf))
if err != nil {
return err
}
if OnFrameErr != nil {
return OnFrameErr
}
2023-11-18 02:57:49 +00:00
if req.Status != nil {
req.Status.DrawProgressBar(len(req.TsFileList), idx)
req.Status.SpeedAddBytes(len(buf))
}
2022-06-19 23:32:53 +00:00
}
err = muxer.WriteTrailer()
if err != nil {
return err
}
2022-06-21 14:17:26 +00:00
err = mp4file.Sync()
if err != nil {
return err
}
2023-11-18 02:57:49 +00:00
if req.Status != nil {
req.Status.DrawProgressBar(1, 1)
2022-08-02 23:37:54 +00:00
}
2022-06-21 14:17:26 +00:00
return nil
2022-06-19 23:32:53 +00:00
}