Files
plugin-rtsp/client.go
2023-02-03 21:36:33 +08:00

154 lines
3.3 KiB
Go

package rtsp
import (
"github.com/aler9/gortsplib/v2"
"github.com/aler9/gortsplib/v2/pkg/url"
"go.uber.org/zap"
"m7s.live/engine/v4"
)
type RTSPPuller struct {
RTSPPublisher
engine.Puller
*gortsplib.Client `json:"-"`
gortsplib.Transport
}
func (p *RTSPPuller) Connect() error {
switch rtspConfig.PullProtocol {
case "tcp", "TCP":
p.Transport = gortsplib.TransportTCP
case "udp", "UDP":
p.Transport = gortsplib.TransportUDP
default:
if p.Transport == gortsplib.TransportTCP {
p.Transport = gortsplib.TransportUDP
} else {
p.Transport = gortsplib.TransportTCP
}
}
p.Client = &gortsplib.Client{
// OnPacketRTP: func(ctx *gortsplib.ClientOnPacketRTPCtx) {
// if p.RTSPPublisher.Tracks[ctx.TrackID] != nil {
// p.RTSPPublisher.Tracks[ctx.TrackID].WriteRTPPack(ctx.Packet)
// }
// },
ReadBufferCount: rtspConfig.ReadBufferSize,
Transport: &p.Transport,
}
// parse URL
u, err := url.Parse(p.RemoteURL)
if err != nil {
return err
}
// connect to the server
if err = p.Client.Start(u.Scheme, u.Host); err != nil {
return err
}
p.SetIO(p.Client)
return nil
}
func (p *RTSPPuller) Pull() (err error) {
u, _ := url.Parse(p.RemoteURL)
defer p.Stop()
if _, err = p.Options(u); err != nil {
p.Error("Options", zap.Error(err))
return
}
// find published tracks
tracks, baseURL, _, err := p.Describe(u)
if err != nil {
p.Error("Describe", zap.Error(err))
return
}
p.tracks = tracks
err = p.SetTracks()
if err != nil {
p.Error("SetTracks", zap.Error(err))
return
}
if err = p.SetupAll(tracks, baseURL); err != nil {
p.Error("SetupAndPlay", zap.Error(err))
return
}
p.OnPacketRTPAny(p.OnPacket)
_, err = p.Play(nil)
if err != nil {
p.Error("Play", zap.Error(err))
return
}
return p.Wait()
}
type RTSPPusher struct {
RTSPSubscriber
engine.Pusher
*gortsplib.Client
gortsplib.Transport
}
func (p *RTSPPusher) OnEvent(event any) {
switch v := event.(type) {
case engine.VideoRTP:
p.Client.WritePacketRTP(p.videoTrack, &v.Packet)
case engine.AudioRTP:
p.Client.WritePacketRTP(p.audioTrack, &v.Packet)
default:
p.RTSPSubscriber.OnEvent(event)
}
}
func (p *RTSPPusher) Connect() error {
if p.Transport == gortsplib.TransportTCP {
p.Transport = gortsplib.TransportUDP
} else {
p.Transport = gortsplib.TransportTCP
}
p.Client = &gortsplib.Client{
ReadBufferCount: rtspConfig.ReadBufferSize,
Transport: &p.Transport,
}
// parse URL
u, err := url.Parse(p.RemoteURL)
if err != nil {
p.Error("url.Parse", zap.Error(err))
return err
}
// connect to the server
if err = p.Client.Start(u.Scheme, u.Host); err != nil {
p.Error("Client.Start", zap.Error(err))
return err
}
p.SetIO(p.Client)
_, err = p.Client.Options(u)
return err
}
func (p *RTSPPusher) Push() (err error) {
var u *url.URL
u, err = url.Parse(p.RemoteURL)
defer p.Stop()
// startTime := time.Now()
// for len(p.tracks) < 2 {
// if time.Sleep(time.Second); time.Since(startTime) > time.Second*10 {
// return fmt.Errorf("timeout")
// }
// }
if _, err = p.Announce(u, p.tracks); err != nil {
p.Error("Announce", zap.Error(err))
return
}
err = p.SetupAll(p.tracks, u)
if err != nil {
p.Error("Setup", zap.Error(err))
return
}
if _, err = p.Record(); err != nil {
p.Error("Record", zap.Error(err))
return
}
p.PlayRTP()
return
}