mirror of
https://github.com/aler9/gortsplib
synced 2025-09-26 19:21:20 +08:00

* switch from v4 to v5 * remove deprecated entities * remove "2" suffix from entities * rename TransportProtocol into Protocol
141 lines
3.1 KiB
Go
141 lines
3.1 KiB
Go
//go:build cgo
|
|
|
|
// Package main contains an example.
|
|
package main
|
|
|
|
import (
|
|
"crypto/rand"
|
|
"log"
|
|
"time"
|
|
|
|
"github.com/bluenviron/gortsplib/v5"
|
|
"github.com/bluenviron/gortsplib/v5/pkg/description"
|
|
"github.com/bluenviron/gortsplib/v5/pkg/format"
|
|
"github.com/bluenviron/gortsplib/v5/pkg/format/rtpsimpleaudio"
|
|
"github.com/bluenviron/mediacommon/v2/pkg/codecs/opus"
|
|
"github.com/pion/rtp"
|
|
)
|
|
|
|
// This example shows how to:
|
|
// 1. connect to a RTSP server, announce a Opus format.
|
|
// 2. generate dummy LPCM audio samples.
|
|
// 3. encode audio samples with Opus.
|
|
// 3. generate RTP packets from Opus packets.
|
|
// 4. write RTP packets to the server.
|
|
|
|
// This example requires the FFmpeg libraries, that can be installed with this command:
|
|
// apt install -y libavcodec-dev gcc pkg-config
|
|
|
|
func multiplyAndDivide(v, m, d int64) int64 {
|
|
secs := v / d
|
|
dec := v % d
|
|
return (secs*m + dec*m/d)
|
|
}
|
|
|
|
func randUint32() (uint32, error) {
|
|
var b [4]byte
|
|
_, err := rand.Read(b[:])
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return uint32(b[0])<<24 | uint32(b[1])<<16 | uint32(b[2])<<8 | uint32(b[3]), nil
|
|
}
|
|
|
|
func encodeMultiple(rtpEnc *rtpsimpleaudio.Encoder, opusPkts [][]byte) ([]*rtp.Packet, error) {
|
|
ret := make([]*rtp.Packet, len(opusPkts))
|
|
pts := uint32(0)
|
|
|
|
for i, opusPkt := range opusPkts {
|
|
var err error
|
|
ret[i], err = rtpEnc.Encode(opusPkt)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ret[i].Timestamp += pts
|
|
|
|
pts += uint32(opus.PacketDuration(opusPkt))
|
|
}
|
|
return ret, nil
|
|
}
|
|
|
|
func main() {
|
|
// create a description that contains a Opus format
|
|
forma := &format.Opus{
|
|
PayloadTyp: 96,
|
|
ChannelCount: 1,
|
|
}
|
|
desc := &description.Session{
|
|
Medias: []*description.Media{{
|
|
Type: description.MediaTypeAudio,
|
|
Formats: []format.Format{forma},
|
|
}},
|
|
}
|
|
|
|
// connect to the server and start recording
|
|
c := gortsplib.Client{}
|
|
err := c.StartRecording("rtsp://myuser:mypass@localhost:8554/mystream", desc)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
defer c.Close()
|
|
|
|
// setup LPCM -> Opus encoder
|
|
opusEnc := &opusEncoder{}
|
|
err = opusEnc.initialize()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// setup Opus -> RTP encoder
|
|
rtpEnc, err := forma.CreateEncoder()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
start := time.Now()
|
|
prevPTS := int64(0)
|
|
|
|
randomStart, err := randUint32()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// setup a ticker to sleep between writings
|
|
ticker := time.NewTicker(100 * time.Millisecond)
|
|
defer ticker.Stop()
|
|
|
|
for range ticker.C {
|
|
// get current timestamp
|
|
pts := multiplyAndDivide(int64(time.Since(start)), int64(forma.ClockRate()), int64(time.Second))
|
|
|
|
// generate dummy LPCM audio samples
|
|
samples := createDummyAudio(pts, prevPTS)
|
|
|
|
// encode samples with Opus
|
|
opusPkts, outPTS, err := opusEnc.encode(samples)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// generate RTP packets from Opus packets
|
|
pkts, err := encodeMultiple(rtpEnc, opusPkts)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
log.Printf("writing RTP packets with PTS=%d, packet count=%d", outPTS, len(pkts))
|
|
|
|
for _, pkt := range pkts {
|
|
pkt.Timestamp += uint32(int64(randomStart) + outPTS)
|
|
|
|
err = c.WritePacketRTP(desc.Medias[0], pkt)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
prevPTS = pts
|
|
}
|
|
}
|