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/rtph265" "github.com/pion/rtp" ) // This example shows how to // 1. connect to a RTSP server // 2. check if there's an H265 media stream // 3. decode the H264 media stream into RGBA frames // This example requires the FFmpeg libraries, that can be installed with this command: // apt install -y libavformat-dev libswscale-dev gcc pkg-config func main() { c := gortsplib.Client{} // parse URL u, err := base.ParseURL("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 available medias desc, _, err := c.Describe(u) if err != nil { panic(err) } // find the H265 media and format var forma *format.H265 medi := desc.FindFormat(&forma) if medi == nil { panic("media not found") } // setup RTP/H265 -> H265 decoder rtpDec, err := forma.CreateDecoder() if err != nil { panic(err) } // setup H265 -> raw frames decoder frameDec := &h265Decoder{} err = frameDec.initialize() if err != nil { panic(err) } defer frameDec.close() // if VPS, SPS and PPS are present into the SDP, send them to the decoder if forma.VPS != nil { frameDec.decode(forma.VPS) } if forma.SPS != nil { frameDec.decode(forma.SPS) } if forma.PPS != nil { frameDec.decode(forma.PPS) } // setup a single media _, err = c.Setup(desc.BaseURL, medi, 0, 0) if err != nil { panic(err) } // 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 != rtph265.ErrNonStartingPacketAndNoPrevious && err != rtph265.ErrMorePacketsNeeded { log.Printf("ERR: %v", err) } return } for _, nalu := range au { // convert NALUs into RGBA frames img, err := frameDec.decode(nalu) if err != nil { panic(err) } // wait for a frame if img == nil { continue } 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()) }