golanggohlsrtmpwebrtcmedia-serverobs-studiortcprtmp-proxyrtmp-serverrtprtsprtsp-proxyrtsp-relayrtsp-serversrtstreamingwebrtc-proxy
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
122 lines
2.3 KiB
122 lines
2.3 KiB
package main |
|
|
|
import ( |
|
"fmt" |
|
"net" |
|
"strconv" |
|
|
|
"github.com/pion/sdp" |
|
) |
|
|
|
func parseIpCidrList(in []string) ([]interface{}, error) { |
|
if len(in) == 0 { |
|
return nil, nil |
|
} |
|
|
|
var ret []interface{} |
|
for _, t := range in { |
|
_, ipnet, err := net.ParseCIDR(t) |
|
if err == nil { |
|
ret = append(ret, ipnet) |
|
continue |
|
} |
|
|
|
ip := net.ParseIP(t) |
|
if ip != nil { |
|
ret = append(ret, ip) |
|
continue |
|
} |
|
|
|
return nil, fmt.Errorf("unable to parse ip/network '%s'", t) |
|
} |
|
return ret, nil |
|
} |
|
|
|
func ipEqualOrInRange(ip net.IP, ips []interface{}) bool { |
|
for _, item := range ips { |
|
switch titem := item.(type) { |
|
case net.IP: |
|
if titem.Equal(ip) { |
|
return true |
|
} |
|
|
|
case *net.IPNet: |
|
if titem.Contains(ip) { |
|
return true |
|
} |
|
} |
|
} |
|
return false |
|
} |
|
|
|
type doubleBuffer struct { |
|
buf1 []byte |
|
buf2 []byte |
|
curBuf bool |
|
} |
|
|
|
func newDoubleBuffer(size int) *doubleBuffer { |
|
return &doubleBuffer{ |
|
buf1: make([]byte, size), |
|
buf2: make([]byte, size), |
|
} |
|
} |
|
|
|
func (db *doubleBuffer) swap() []byte { |
|
var ret []byte |
|
if !db.curBuf { |
|
ret = db.buf1 |
|
} else { |
|
ret = db.buf2 |
|
} |
|
db.curBuf = !db.curBuf |
|
return ret |
|
} |
|
|
|
func sdpForServer(sin *sdp.SessionDescription) (*sdp.SessionDescription, []byte) { |
|
sout := &sdp.SessionDescription{ |
|
SessionName: "Stream", |
|
Origin: sdp.Origin{ |
|
Username: "-", |
|
NetworkType: "IN", |
|
AddressType: "IP4", |
|
UnicastAddress: "127.0.0.1", |
|
}, |
|
TimeDescriptions: []sdp.TimeDescription{ |
|
{Timing: sdp.Timing{0, 0}}, |
|
}, |
|
} |
|
|
|
for i, min := range sin.MediaDescriptions { |
|
mout := &sdp.MediaDescription{ |
|
MediaName: sdp.MediaName{ |
|
Media: min.MediaName.Media, |
|
Protos: []string{"RTP", "AVP"}, // override protocol |
|
Formats: min.MediaName.Formats, |
|
}, |
|
Bandwidth: min.Bandwidth, |
|
Attributes: func() []sdp.Attribute { |
|
var ret []sdp.Attribute |
|
|
|
for _, attr := range min.Attributes { |
|
if attr.Key == "rtpmap" || attr.Key == "fmtp" { |
|
ret = append(ret, attr) |
|
} |
|
} |
|
|
|
// control attribute is mandatory, and is the path that is appended |
|
// to the stream path in SETUP |
|
ret = append(ret, sdp.Attribute{ |
|
Key: "control", |
|
Value: "trackID=" + strconv.FormatInt(int64(i), 10), |
|
}) |
|
|
|
return ret |
|
}(), |
|
} |
|
sout.MediaDescriptions = append(sout.MediaDescriptions, mout) |
|
} |
|
|
|
bytsout := []byte(sout.Marshal()) |
|
return sout, bytsout |
|
}
|
|
|