Files
gortsplib/examples/client-play-format-h264/main.go
2025-09-15 19:00:50 +02:00

136 lines
2.8 KiB
Go

//go:build cgo
// Package main contains an example.
package main
import (
"log"
"github.com/bluenviron/gortsplib/v4"
"github.com/bluenviron/gortsplib/v4/pkg/base"
"github.com/bluenviron/gortsplib/v4/pkg/format"
"github.com/bluenviron/gortsplib/v4/pkg/format/rtph264"
"github.com/bluenviron/mediacommon/v2/pkg/codecs/h264"
"github.com/pion/rtp"
)
// This example shows how to:
// 1. connect to a RTSP server.
// 2. check if there's an H264 stream.
// 3. decode the H264 stream into RGBA frames.
// This example requires the FFmpeg libraries, that can be installed with this command:
// apt install -y libavcodec-dev libswscale-dev gcc pkg-config
func main() {
// parse URL
u, err := base.ParseURL("rtsp://myuser:mypass@localhost:8554/mystream")
if err != nil {
panic(err)
}
c := gortsplib.Client{
Scheme: u.Scheme,
Host: u.Host,
}
// connect to the server
err = c.Start2()
if err != nil {
panic(err)
}
defer c.Close()
// find available medias
desc, _, err := c.Describe(u)
if err != nil {
panic(err)
}
// find the H264 media and format
var forma *format.H264
medi := desc.FindFormat(&forma)
if medi == nil {
panic("media not found")
}
// setup RTP -> H264 decoder
rtpDec, err := forma.CreateDecoder()
if err != nil {
panic(err)
}
// setup H264 -> RGBA decoder
h264Dec := &h264Decoder{}
err = h264Dec.initialize()
if err != nil {
panic(err)
}
defer h264Dec.close()
// if SPS and PPS are present into the SDP, send them to the decoder
if forma.SPS != nil {
h264Dec.decode([][]byte{forma.SPS})
}
if forma.PPS != nil {
h264Dec.decode([][]byte{forma.PPS})
}
// setup a single media
_, err = c.Setup(desc.BaseURL, medi, 0, 0)
if err != nil {
panic(err)
}
firstRandomAccess := false
// called when a RTP packet arrives
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
// decode timestamp
pts, ok := c.PacketPTS2(medi, pkt)
if !ok {
log.Printf("waiting for timestamp")
return
}
// extract access units from RTP packets
au, err := rtpDec.Decode(pkt)
if err != nil {
if err != rtph264.ErrNonStartingPacketAndNoPrevious && err != rtph264.ErrMorePacketsNeeded {
log.Printf("ERR: %v", err)
}
return
}
// wait for a random access unit
if !firstRandomAccess && !h264.IsRandomAccess(au) {
log.Printf("waiting for a random access unit")
return
}
firstRandomAccess = true
// convert H264 access units into RGBA frames
img, err := h264Dec.decode(au)
if err != nil {
panic(err)
}
// check for frame presence
if img == nil {
log.Printf("ERR: frame cannot be decoded")
return
}
log.Printf("decoded frame with PTS %v and size %v", pts, img.Bounds().Max)
})
// start playing
_, err = c.Play(nil)
if err != nil {
panic(err)
}
// wait until a fatal error
panic(c.Wait())
}