mirror of
https://github.com/aler9/gortsplib
synced 2025-10-24 07:34:17 +08:00
160 lines
3.1 KiB
Go
160 lines
3.1 KiB
Go
package main
|
|
|
|
import (
|
|
"image"
|
|
"image/jpeg"
|
|
"log"
|
|
"os"
|
|
"strconv"
|
|
"time"
|
|
|
|
"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/pkg/codecs/h264"
|
|
"github.com/pion/rtp"
|
|
)
|
|
|
|
// This example shows how to
|
|
// 1. connect to a RTSP server
|
|
// 2. check if there's a H264 format
|
|
// 3. decode the H264 format into RGBA frames
|
|
// 4. convert frames to JPEG images and save them on disk
|
|
|
|
// This example requires the FFmpeg libraries, that can be installed with this command:
|
|
// apt install -y libavformat-dev libswscale-dev gcc pkg-config
|
|
|
|
func saveToFile(img image.Image) error {
|
|
// create file
|
|
fname := strconv.FormatInt(time.Now().UnixNano()/int64(time.Millisecond), 10) + ".jpg"
|
|
f, err := os.Create(fname)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
defer f.Close()
|
|
|
|
log.Println("saving", fname)
|
|
|
|
// convert to jpeg
|
|
return jpeg.Encode(f, img, &jpeg.Options{
|
|
Quality: 60,
|
|
})
|
|
}
|
|
|
|
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 H264 media and format
|
|
var forma *format.H264
|
|
medi := desc.FindFormat(&forma)
|
|
if medi == nil {
|
|
panic("media not found")
|
|
}
|
|
|
|
// setup RTP/H264 -> H264 decoder
|
|
rtpDec, err := forma.CreateDecoder()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// setup H264 -> raw frames decoder
|
|
frameDec := &h264Decoder{}
|
|
err = frameDec.initialize()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
defer frameDec.close()
|
|
|
|
// if SPS and PPS are present into the SDP, send them to the decoder
|
|
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)
|
|
}
|
|
|
|
iframeReceived := false
|
|
saveCount := 0
|
|
|
|
// called when a RTP packet arrives
|
|
c.OnPacketRTP(medi, forma, func(pkt *rtp.Packet) {
|
|
// 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 an I-frame
|
|
if !iframeReceived {
|
|
if !h264.IDRPresent(au) {
|
|
log.Printf("waiting for an I-frame")
|
|
return
|
|
}
|
|
iframeReceived = true
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
// convert frame to JPEG and save to file
|
|
err = saveToFile(img)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
saveCount++
|
|
if saveCount == 5 {
|
|
log.Printf("saved 5 images, exiting")
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
})
|
|
|
|
// start playing
|
|
_, err = c.Play(nil)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// wait until a fatal error
|
|
panic(c.Wait())
|
|
}
|