//go:build cgo // Package main contains an example. package main import ( "log" "github.com/bluenviron/gortsplib/v5" "github.com/bluenviron/gortsplib/v5/pkg/base" "github.com/bluenviron/gortsplib/v5/pkg/format" "github.com/bluenviron/gortsplib/v5/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.Start() 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.PacketPTS(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()) }