mirror of
https://github.com/aler9/gortsplib
synced 2025-10-05 07:06:58 +08:00
add new example client-read-format-mpeg4audio-save-to-disk (#235)
This commit is contained in:
@@ -70,6 +70,7 @@ Features:
|
|||||||
* [client-read-format-lpcm](examples/client-read-format-lpcm/main.go)
|
* [client-read-format-lpcm](examples/client-read-format-lpcm/main.go)
|
||||||
* [client-read-format-mjpeg](examples/client-read-format-mjpeg/main.go)
|
* [client-read-format-mjpeg](examples/client-read-format-mjpeg/main.go)
|
||||||
* [client-read-format-mpeg4audio](examples/client-read-format-mpeg4audio/main.go)
|
* [client-read-format-mpeg4audio](examples/client-read-format-mpeg4audio/main.go)
|
||||||
|
* [client-read-format-mpeg4audio-save-to-disk](examples/client-read-format-mpeg4audio-save-to-disk/main.go)
|
||||||
* [client-read-format-opus](examples/client-read-format-opus/main.go)
|
* [client-read-format-opus](examples/client-read-format-opus/main.go)
|
||||||
* [client-read-format-vp8](examples/client-read-format-vp8/main.go)
|
* [client-read-format-vp8](examples/client-read-format-vp8/main.go)
|
||||||
* [client-read-format-vp9](examples/client-read-format-vp9/main.go)
|
* [client-read-format-vp9](examples/client-read-format-vp9/main.go)
|
||||||
|
@@ -97,8 +97,8 @@ func main() {
|
|||||||
// called when a RTP packet arrives
|
// called when a RTP packet arrives
|
||||||
saveCount := 0
|
saveCount := 0
|
||||||
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
||||||
// extract NALUs from RTP packets
|
// extract access units from RTP packets
|
||||||
nalus, _, err := rtpDec.Decode(pkt)
|
au, _, err := rtpDec.Decode(pkt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if err != rtph264.ErrNonStartingPacketAndNoPrevious && err != rtph264.ErrMorePacketsNeeded {
|
if err != rtph264.ErrNonStartingPacketAndNoPrevious && err != rtph264.ErrMorePacketsNeeded {
|
||||||
log.Printf("ERR: %v", err)
|
log.Printf("ERR: %v", err)
|
||||||
@@ -106,7 +106,7 @@ func main() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, nalu := range nalus {
|
for _, nalu := range au {
|
||||||
// convert NALUs into RGBA frames
|
// convert NALUs into RGBA frames
|
||||||
img, err := h264RawDec.decode(nalu)
|
img, err := h264RawDec.decode(nalu)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@@ -13,7 +13,7 @@ import (
|
|||||||
// This example shows how to
|
// This example shows how to
|
||||||
// 1. connect to a RTSP server
|
// 1. connect to a RTSP server
|
||||||
// 2. check if there's a H264 media
|
// 2. check if there's a H264 media
|
||||||
// 3. save the content of the H264 media into a file in MPEG-TS format
|
// 3. save the content of the media into a file in MPEG-TS format
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
c := gortsplib.Client{}
|
c := gortsplib.Client{}
|
||||||
@@ -47,7 +47,7 @@ func main() {
|
|||||||
// setup RTP/H264 -> H264 decoder
|
// setup RTP/H264 -> H264 decoder
|
||||||
rtpDec := forma.CreateDecoder()
|
rtpDec := forma.CreateDecoder()
|
||||||
|
|
||||||
// setup H264->MPEGTS muxer
|
// setup H264 -> MPEG-TS muxer
|
||||||
mpegtsMuxer, err := newMPEGTSMuxer(forma.SPS, forma.PPS)
|
mpegtsMuxer, err := newMPEGTSMuxer(forma.SPS, forma.PPS)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
@@ -61,9 +61,9 @@ func main() {
|
|||||||
|
|
||||||
// called when a RTP packet arrives
|
// called when a RTP packet arrives
|
||||||
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
||||||
// extract NALUs from RTP packets
|
// extract access unit from RTP packets
|
||||||
// DecodeUntilMarker is necessary for the DTS extractor to work
|
// DecodeUntilMarker is necessary for the DTS extractor to work
|
||||||
nalus, pts, err := rtpDec.DecodeUntilMarker(pkt)
|
au, pts, err := rtpDec.DecodeUntilMarker(pkt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if err != rtph264.ErrNonStartingPacketAndNoPrevious && err != rtph264.ErrMorePacketsNeeded {
|
if err != rtph264.ErrNonStartingPacketAndNoPrevious && err != rtph264.ErrMorePacketsNeeded {
|
||||||
log.Printf("ERR: %v", err)
|
log.Printf("ERR: %v", err)
|
||||||
@@ -71,8 +71,8 @@ func main() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// encode H264 NALUs into MPEG-TS
|
// encode the access unit into MPEG-TS
|
||||||
mpegtsMuxer.encode(nalus, pts)
|
mpegtsMuxer.encode(au, pts)
|
||||||
})
|
})
|
||||||
|
|
||||||
// start playing
|
// start playing
|
||||||
|
@@ -54,8 +54,8 @@ func (e *mpegtsMuxer) close() {
|
|||||||
e.f.Close()
|
e.f.Close()
|
||||||
}
|
}
|
||||||
|
|
||||||
// encode encodes H264 NALUs into MPEG-TS.
|
// encode encodes a H264 access unit into MPEG-TS.
|
||||||
func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
func (e *mpegtsMuxer) encode(au [][]byte, pts time.Duration) error {
|
||||||
// prepend an AUD. This is required by some players
|
// prepend an AUD. This is required by some players
|
||||||
filteredNALUs := [][]byte{
|
filteredNALUs := [][]byte{
|
||||||
{byte(h264.NALUTypeAccessUnitDelimiter), 240},
|
{byte(h264.NALUTypeAccessUnitDelimiter), 240},
|
||||||
@@ -64,7 +64,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
nonIDRPresent := false
|
nonIDRPresent := false
|
||||||
idrPresent := false
|
idrPresent := false
|
||||||
|
|
||||||
for _, nalu := range nalus {
|
for _, nalu := range au {
|
||||||
typ := h264.NALUType(nalu[0] & 0x1F)
|
typ := h264.NALUType(nalu[0] & 0x1F)
|
||||||
switch typ {
|
switch typ {
|
||||||
case h264.NALUTypeSPS:
|
case h264.NALUTypeSPS:
|
||||||
@@ -88,7 +88,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
filteredNALUs = append(filteredNALUs, nalu)
|
filteredNALUs = append(filteredNALUs, nalu)
|
||||||
}
|
}
|
||||||
|
|
||||||
nalus = filteredNALUs
|
au = filteredNALUs
|
||||||
|
|
||||||
if !nonIDRPresent && !idrPresent {
|
if !nonIDRPresent && !idrPresent {
|
||||||
return nil
|
return nil
|
||||||
@@ -96,7 +96,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
|
|
||||||
// add SPS and PPS before every group that contains an IDR
|
// add SPS and PPS before every group that contains an IDR
|
||||||
if idrPresent {
|
if idrPresent {
|
||||||
nalus = append([][]byte{e.sps, e.pps}, nalus...)
|
au = append([][]byte{e.sps, e.pps}, au...)
|
||||||
}
|
}
|
||||||
|
|
||||||
var dts time.Duration
|
var dts time.Duration
|
||||||
@@ -111,7 +111,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
e.dtsExtractor = h264.NewDTSExtractor()
|
e.dtsExtractor = h264.NewDTSExtractor()
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
dts, err = e.dtsExtractor.Extract(nalus, pts)
|
dts, err = e.dtsExtractor.Extract(au, pts)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -122,7 +122,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
|
|
||||||
} else {
|
} else {
|
||||||
var err error
|
var err error
|
||||||
dts, err = e.dtsExtractor.Extract(nalus, pts)
|
dts, err = e.dtsExtractor.Extract(au, pts)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -145,7 +145,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// encode into Annex-B
|
// encode into Annex-B
|
||||||
annexb, err := h264.AnnexBMarshal(nalus)
|
annexb, err := h264.AnnexBMarshal(au)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@@ -73,8 +73,8 @@ func main() {
|
|||||||
|
|
||||||
// called when a RTP packet arrives
|
// called when a RTP packet arrives
|
||||||
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
||||||
// extract NALUs from RTP packets
|
// extract access units from RTP packets
|
||||||
nalus, pts, err := rtpDec.Decode(pkt)
|
au, pts, err := rtpDec.Decode(pkt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if err != rtph264.ErrNonStartingPacketAndNoPrevious && err != rtph264.ErrMorePacketsNeeded {
|
if err != rtph264.ErrNonStartingPacketAndNoPrevious && err != rtph264.ErrMorePacketsNeeded {
|
||||||
log.Printf("ERR: %v", err)
|
log.Printf("ERR: %v", err)
|
||||||
@@ -82,7 +82,7 @@ func main() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, nalu := range nalus {
|
for _, nalu := range au {
|
||||||
// convert NALUs into RGBA frames
|
// convert NALUs into RGBA frames
|
||||||
img, err := h264RawDec.decode(nalu)
|
img, err := h264RawDec.decode(nalu)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@@ -55,8 +55,8 @@ func main() {
|
|||||||
|
|
||||||
// called when a RTP packet arrives
|
// called when a RTP packet arrives
|
||||||
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
||||||
// extract NALUs from RTP packets
|
// extract access units from RTP packets
|
||||||
nalus, pts, err := rtpDec.Decode(pkt)
|
au, pts, err := rtpDec.Decode(pkt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if err != rtph265.ErrNonStartingPacketAndNoPrevious && err != rtph265.ErrMorePacketsNeeded {
|
if err != rtph265.ErrNonStartingPacketAndNoPrevious && err != rtph265.ErrMorePacketsNeeded {
|
||||||
log.Printf("ERR: %v", err)
|
log.Printf("ERR: %v", err)
|
||||||
@@ -64,7 +64,7 @@ func main() {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, nalu := range nalus {
|
for _, nalu := range au {
|
||||||
log.Printf("received NALU with PTS %v and size %d\n", pts, len(nalu))
|
log.Printf("received NALU with PTS %v and size %d\n", pts, len(nalu))
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
84
examples/client-read-format-mpeg4audio-save-to-disk/main.go
Normal file
84
examples/client-read-format-mpeg4audio-save-to-disk/main.go
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
|
||||||
|
"github.com/bluenviron/gortsplib/v3"
|
||||||
|
"github.com/bluenviron/gortsplib/v3/pkg/formats"
|
||||||
|
"github.com/bluenviron/gortsplib/v3/pkg/url"
|
||||||
|
"github.com/pion/rtp"
|
||||||
|
)
|
||||||
|
|
||||||
|
// This example shows how to
|
||||||
|
// 1. connect to a RTSP server
|
||||||
|
// 2. check if there's an MPEG4-audio media
|
||||||
|
// 3. save the content of the media into a file in MPEG-TS format
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
c := gortsplib.Client{}
|
||||||
|
|
||||||
|
// parse URL
|
||||||
|
u, err := url.Parse("rtsp://localhost:8554/mystream")
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// connect to the server
|
||||||
|
err = c.Start(u.Scheme, u.Host)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
defer c.Close()
|
||||||
|
|
||||||
|
// find published medias
|
||||||
|
medias, baseURL, _, err := c.Describe(u)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// find the MPEG4-audio media and format
|
||||||
|
var forma *formats.MPEG4Audio
|
||||||
|
medi := medias.FindFormat(&forma)
|
||||||
|
if medi == nil {
|
||||||
|
panic("media not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
// setup RTP/MPEG4-audio -> MPEG4-audio decoder
|
||||||
|
rtpDec := forma.CreateDecoder()
|
||||||
|
|
||||||
|
// setup MPEG4-audio -> MPEG-TS muxer
|
||||||
|
mpegtsMuxer, err := newMPEGTSMuxer(forma.Config)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// setup a single media
|
||||||
|
_, err = c.Setup(medi, baseURL, 0, 0)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// called when a RTP packet arrives
|
||||||
|
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
||||||
|
// extract access units from RTP packets
|
||||||
|
aus, pts, err := rtpDec.Decode(pkt)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("ERR: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, au := range aus {
|
||||||
|
// encode the access unit into MPEG-TS
|
||||||
|
mpegtsMuxer.encode(au, pts)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// start playing
|
||||||
|
_, err = c.Play(nil)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// wait until a fatal error
|
||||||
|
panic(c.Wait())
|
||||||
|
}
|
@@ -0,0 +1,91 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"context"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/asticode/go-astits"
|
||||||
|
"github.com/bluenviron/mediacommon/pkg/codecs/mpeg4audio"
|
||||||
|
)
|
||||||
|
|
||||||
|
// mpegtsMuxer allows to save a MPEG4-audio stream into a MPEG-TS file.
|
||||||
|
type mpegtsMuxer struct {
|
||||||
|
config *mpeg4audio.Config
|
||||||
|
f *os.File
|
||||||
|
b *bufio.Writer
|
||||||
|
mux *astits.Muxer
|
||||||
|
}
|
||||||
|
|
||||||
|
// newMPEGTSMuxer allocates a mpegtsMuxer.
|
||||||
|
func newMPEGTSMuxer(config *mpeg4audio.Config) (*mpegtsMuxer, error) {
|
||||||
|
f, err := os.Create("mystream.ts")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
b := bufio.NewWriter(f)
|
||||||
|
|
||||||
|
mux := astits.NewMuxer(context.Background(), b)
|
||||||
|
mux.AddElementaryStream(astits.PMTElementaryStream{
|
||||||
|
ElementaryPID: 257,
|
||||||
|
StreamType: astits.StreamTypeAACAudio,
|
||||||
|
})
|
||||||
|
mux.SetPCRPID(257)
|
||||||
|
|
||||||
|
return &mpegtsMuxer{
|
||||||
|
config: config,
|
||||||
|
f: f,
|
||||||
|
b: b,
|
||||||
|
mux: mux,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// close closes all the mpegtsMuxer resources.
|
||||||
|
func (e *mpegtsMuxer) close() {
|
||||||
|
e.b.Flush()
|
||||||
|
e.f.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
// encode encodes a MPEG4-audio access unit into MPEG-TS.
|
||||||
|
func (e *mpegtsMuxer) encode(au []byte, pts time.Duration) error {
|
||||||
|
// wrap access unit inside an ADTS packet
|
||||||
|
pkts := mpeg4audio.ADTSPackets{
|
||||||
|
{
|
||||||
|
Type: e.config.Type,
|
||||||
|
SampleRate: e.config.SampleRate,
|
||||||
|
ChannelCount: e.config.ChannelCount,
|
||||||
|
AU: au,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
enc, err := pkts.Marshal()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err = e.mux.WriteData(&astits.MuxerData{
|
||||||
|
PID: 257,
|
||||||
|
AdaptationField: &astits.PacketAdaptationField{
|
||||||
|
RandomAccessIndicator: true,
|
||||||
|
},
|
||||||
|
PES: &astits.PESData{
|
||||||
|
Header: &astits.PESHeader{
|
||||||
|
OptionalHeader: &astits.PESOptionalHeader{
|
||||||
|
MarkerBits: 2,
|
||||||
|
PTSDTSIndicator: astits.PTSDTSIndicatorOnlyPTS,
|
||||||
|
PTS: &astits.ClockReference{Base: int64(pts.Seconds() * 90000)},
|
||||||
|
},
|
||||||
|
PacketLength: uint16(len(enc) + 8),
|
||||||
|
StreamID: 192, // audio
|
||||||
|
},
|
||||||
|
Data: enc,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Println("wrote TS packet")
|
||||||
|
return nil
|
||||||
|
}
|
Reference in New Issue
Block a user