implement Dialer

This commit is contained in:
aler9
2020-11-01 20:02:42 +01:00
parent 812545b776
commit 98b521b058
3 changed files with 167 additions and 112 deletions

View File

@@ -1,108 +0,0 @@
package gortsplib
import (
"github.com/aler9/gortsplib/base"
)
// DialRead connects to the address and starts reading all tracks.
func DialRead(address string, proto StreamProtocol) (*ConnClient, error) {
u, err := base.ParseURL(address)
if err != nil {
return nil, err
}
conn, err := NewConnClient(ConnClientConf{Host: u.Host()})
if err != nil {
return nil, err
}
_, err = conn.Options(u)
if err != nil {
conn.Close()
return nil, err
}
tracks, _, err := conn.Describe(u)
if err != nil {
conn.Close()
return nil, err
}
if proto == StreamProtocolUDP {
for _, track := range tracks {
_, err := conn.SetupUDP(u, TransportModePlay, track, 0, 0)
if err != nil {
return nil, err
}
}
} else {
for _, track := range tracks {
_, err := conn.SetupTCP(u, TransportModePlay, track)
if err != nil {
conn.Close()
return nil, err
}
}
}
_, err = conn.Play(u)
if err != nil {
conn.Close()
return nil, err
}
return conn, nil
}
// DialPublish connects to the address and starts publishing the tracks.
func DialPublish(address string, proto StreamProtocol, tracks Tracks) (*ConnClient, error) {
u, err := base.ParseURL(address)
if err != nil {
return nil, err
}
conn, err := NewConnClient(ConnClientConf{Host: u.Host()})
if err != nil {
return nil, err
}
_, err = conn.Options(u)
if err != nil {
conn.Close()
return nil, err
}
_, err = conn.Announce(u, tracks)
if err != nil {
conn.Close()
return nil, err
}
if proto == StreamProtocolUDP {
for _, track := range tracks {
_, err = conn.SetupUDP(u, TransportModeRecord, track, 0, 0)
if err != nil {
conn.Close()
return nil, err
}
}
} else {
for _, track := range tracks {
_, err = conn.SetupTCP(u, TransportModeRecord, track)
if err != nil {
conn.Close()
return nil, err
}
}
}
_, err = conn.Record(u)
if err != nil {
conn.Close()
return nil, err
}
return conn, nil
}

163
dialer.go Normal file
View File

@@ -0,0 +1,163 @@
package gortsplib
import (
"net"
"time"
"github.com/aler9/gortsplib/base"
)
// DefaultDialer is the default dialer, used by DialRead and DialPublish.
var DefaultDialer = &Dialer{}
// DialRead connects to the address and starts reading all tracks.
func DialRead(address string, proto StreamProtocol) (*ConnClient, error) {
return DefaultDialer.DialRead(address, proto)
}
// DialPublish connects to the address and starts publishing the tracks.
func DialPublish(address string, proto StreamProtocol, tracks Tracks) (*ConnClient, error) {
return DefaultDialer.DialPublish(address, proto, tracks)
}
// Dialer allows to connect to a server and read or publish tracks.
type Dialer struct {
// (optional) timeout of read operations.
// It defaults to 10 seconds
ReadTimeout time.Duration
// (optional) timeout of write operations.
// It defaults to 5 seconds
WriteTimeout time.Duration
// (optional) read buffer count.
// If greater than 1, allows to pass buffers to routines different than the one
// that is reading frames.
// It defaults to 1
ReadBufferCount int
// (optional) function used to initialize the TCP client.
// It defaults to net.DialTimeout
DialTimeout func(network, address string, timeout time.Duration) (net.Conn, error)
// (optional) function used to initialize UDP listeners.
// It defaults to net.ListenPacket
ListenPacket func(network, address string) (net.PacketConn, error)
}
// DialRead connects to the address and starts reading all tracks.
func (d *Dialer) DialRead(address string, proto StreamProtocol) (*ConnClient, error) {
u, err := base.ParseURL(address)
if err != nil {
return nil, err
}
conn, err := NewConnClient(ConnClientConf{
Host: u.Host(),
ReadTimeout: d.ReadTimeout,
WriteTimeout: d.WriteTimeout,
ReadBufferCount: d.ReadBufferCount,
DialTimeout: d.DialTimeout,
ListenPacket: d.ListenPacket,
})
if err != nil {
return nil, err
}
_, err = conn.Options(u)
if err != nil {
conn.Close()
return nil, err
}
tracks, _, err := conn.Describe(u)
if err != nil {
conn.Close()
return nil, err
}
if proto == StreamProtocolUDP {
for _, track := range tracks {
_, err := conn.SetupUDP(u, TransportModePlay, track, 0, 0)
if err != nil {
return nil, err
}
}
} else {
for _, track := range tracks {
_, err := conn.SetupTCP(u, TransportModePlay, track)
if err != nil {
conn.Close()
return nil, err
}
}
}
_, err = conn.Play(u)
if err != nil {
conn.Close()
return nil, err
}
return conn, nil
}
// DialPublish connects to the address and starts publishing the tracks.
func (d *Dialer) DialPublish(address string, proto StreamProtocol, tracks Tracks) (*ConnClient, error) {
u, err := base.ParseURL(address)
if err != nil {
return nil, err
}
conn, err := NewConnClient(ConnClientConf{
Host: u.Host(),
ReadTimeout: d.ReadTimeout,
WriteTimeout: d.WriteTimeout,
ReadBufferCount: d.ReadBufferCount,
DialTimeout: d.DialTimeout,
ListenPacket: d.ListenPacket,
})
if err != nil {
return nil, err
}
_, err = conn.Options(u)
if err != nil {
conn.Close()
return nil, err
}
_, err = conn.Announce(u, tracks)
if err != nil {
conn.Close()
return nil, err
}
if proto == StreamProtocolUDP {
for _, track := range tracks {
_, err = conn.SetupUDP(u, TransportModeRecord, track, 0, 0)
if err != nil {
conn.Close()
return nil, err
}
}
} else {
for _, track := range tracks {
_, err = conn.SetupTCP(u, TransportModeRecord, track)
if err != nil {
conn.Close()
return nil, err
}
}
}
_, err = conn.Record(u)
if err != nil {
conn.Close()
return nil, err
}
return conn, nil
}

View File

@@ -58,7 +58,7 @@ func (c *container) wait() int {
return int(code) return int(code)
} }
func TestConnClientDialReadUDP(t *testing.T) { func TestDialerReadUDP(t *testing.T) {
cnt1, err := newContainer("rtsp-simple-server", "server", []string{}) cnt1, err := newContainer("rtsp-simple-server", "server", []string{})
require.NoError(t, err) require.NoError(t, err)
defer cnt1.close() defer cnt1.close()
@@ -97,7 +97,7 @@ func TestConnClientDialReadUDP(t *testing.T) {
conn.CloseUDPListeners() conn.CloseUDPListeners()
} }
func TestConnClientDialReadTCP(t *testing.T) { func TestDialerReadTCP(t *testing.T) {
cnt1, err := newContainer("rtsp-simple-server", "server", []string{}) cnt1, err := newContainer("rtsp-simple-server", "server", []string{})
require.NoError(t, err) require.NoError(t, err)
defer cnt1.close() defer cnt1.close()
@@ -129,7 +129,7 @@ func TestConnClientDialReadTCP(t *testing.T) {
require.Equal(t, StreamTypeRtp, typ) require.Equal(t, StreamTypeRtp, typ)
} }
func TestConnClientDialPublishUDP(t *testing.T) { func TestDialerPublishUDP(t *testing.T) {
for _, server := range []string{ for _, server := range []string{
"rtsp-simple-server", "rtsp-simple-server",
"ffmpeg", "ffmpeg",
@@ -228,7 +228,7 @@ func TestConnClientDialPublishUDP(t *testing.T) {
} }
} }
func TestConnClientDialPublishTCP(t *testing.T) { func TestDialerPublishTCP(t *testing.T) {
for _, server := range []string{ for _, server := range []string{
"rtsp-simple-server", "rtsp-simple-server",
"ffmpeg", "ffmpeg",