mirror of
https://github.com/jech/galene.git
synced 2024-11-09 18:25:58 +01:00
0f7cbe5701
ErrTimeout was introduced in Go 1.15, while we support 1.13 and later. Thanks to Michael Ströder for the report.
252 lines
5.1 KiB
Go
252 lines
5.1 KiB
Go
package ice
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/hmac"
|
|
"crypto/sha1"
|
|
"encoding/base64"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/pion/webrtc/v3"
|
|
|
|
"github.com/jech/galene/turnserver"
|
|
)
|
|
|
|
type timeoutError struct{}
|
|
|
|
func (e timeoutError) Error() string {
|
|
return "timeout"
|
|
}
|
|
|
|
func (e timeoutError) Timeout() bool {
|
|
return true
|
|
}
|
|
|
|
var errTimeout = timeoutError{}
|
|
|
|
type Server struct {
|
|
URLs []string `json:"urls"`
|
|
Username string `json:"username,omitempty"`
|
|
Credential interface{} `json:"credential,omitempty"`
|
|
CredentialType string `json:"credentialType,omitempty"`
|
|
}
|
|
|
|
func getServer(server Server) (webrtc.ICEServer, error) {
|
|
s := webrtc.ICEServer{
|
|
URLs: server.URLs,
|
|
Username: server.Username,
|
|
Credential: server.Credential,
|
|
}
|
|
switch server.CredentialType {
|
|
case "", "password":
|
|
s.CredentialType = webrtc.ICECredentialTypePassword
|
|
case "oauth":
|
|
s.CredentialType = webrtc.ICECredentialTypeOauth
|
|
case "hmac-sha1":
|
|
cred, ok := server.Credential.(string)
|
|
if !ok {
|
|
return webrtc.ICEServer{},
|
|
errors.New("credential is not a string")
|
|
}
|
|
ts := time.Now().Unix() + 86400
|
|
var username string
|
|
if server.Username == "" {
|
|
username = fmt.Sprintf("%d", ts)
|
|
} else {
|
|
username = fmt.Sprintf("%d:%s", ts, server.Username)
|
|
}
|
|
mac := hmac.New(sha1.New, []byte(cred))
|
|
mac.Write([]byte(username))
|
|
buf := bytes.Buffer{}
|
|
e := base64.NewEncoder(base64.StdEncoding, &buf)
|
|
e.Write(mac.Sum(nil))
|
|
e.Close()
|
|
s.Username = username
|
|
s.Credential = string(buf.Bytes())
|
|
s.CredentialType = webrtc.ICECredentialTypePassword
|
|
default:
|
|
return webrtc.ICEServer{}, errors.New("unsupported credential type")
|
|
}
|
|
return s, nil
|
|
}
|
|
|
|
var ICEFilename string
|
|
var ICERelayOnly bool
|
|
|
|
type configuration struct {
|
|
conf webrtc.Configuration
|
|
timestamp time.Time
|
|
}
|
|
|
|
var conf atomic.Value
|
|
|
|
func Update() *configuration {
|
|
now := time.Now()
|
|
var cf webrtc.Configuration
|
|
|
|
found := false
|
|
if ICEFilename != "" {
|
|
found = true
|
|
file, err := os.Open(ICEFilename)
|
|
if err != nil {
|
|
if !os.IsNotExist(err) {
|
|
log.Printf("Open %v: %v", ICEFilename, err)
|
|
} else {
|
|
found = false
|
|
}
|
|
} else {
|
|
defer file.Close()
|
|
d := json.NewDecoder(file)
|
|
var servers []Server
|
|
err = d.Decode(&servers)
|
|
if err != nil {
|
|
log.Printf("Get ICE configuration: %v", err)
|
|
}
|
|
for _, s := range servers {
|
|
ss, err := getServer(s)
|
|
if err != nil {
|
|
log.Printf("parse ICE server: %v", err)
|
|
continue
|
|
}
|
|
cf.ICEServers = append(cf.ICEServers, ss)
|
|
}
|
|
}
|
|
}
|
|
|
|
err := turnserver.StartStop(!found)
|
|
if err != nil {
|
|
log.Printf("TURN: %v", err)
|
|
}
|
|
|
|
cf.ICEServers = append(cf.ICEServers, turnserver.ICEServers()...)
|
|
|
|
if ICERelayOnly {
|
|
cf.ICETransportPolicy = webrtc.ICETransportPolicyRelay
|
|
}
|
|
|
|
iceConf := configuration{
|
|
conf: cf,
|
|
timestamp: now,
|
|
}
|
|
conf.Store(&iceConf)
|
|
return &iceConf
|
|
}
|
|
|
|
func ICEConfiguration() *webrtc.Configuration {
|
|
conf, ok := conf.Load().(*configuration)
|
|
if !ok || time.Since(conf.timestamp) > 5*time.Minute {
|
|
conf = Update()
|
|
} else if time.Since(conf.timestamp) > 2*time.Minute {
|
|
go Update()
|
|
}
|
|
|
|
return &conf.conf
|
|
}
|
|
|
|
func RelayTest(timeout time.Duration) (time.Duration, error) {
|
|
|
|
conf := ICEConfiguration()
|
|
conf2 := *conf
|
|
conf2.ICETransportPolicy = webrtc.ICETransportPolicyRelay
|
|
|
|
var s webrtc.SettingEngine
|
|
s.SetHostAcceptanceMinWait(0)
|
|
s.SetSrflxAcceptanceMinWait(0)
|
|
s.SetPrflxAcceptanceMinWait(0)
|
|
s.SetRelayAcceptanceMinWait(0)
|
|
api := webrtc.NewAPI(webrtc.WithSettingEngine(s))
|
|
|
|
pc1, err := api.NewPeerConnection(conf2)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
defer pc1.Close()
|
|
pc2, err := webrtc.NewPeerConnection(*conf)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
defer pc2.Close()
|
|
|
|
pc1.OnICECandidate(func(c *webrtc.ICECandidate) {
|
|
if c != nil {
|
|
pc2.AddICECandidate(c.ToJSON())
|
|
}
|
|
})
|
|
pc2.OnICECandidate(func(c *webrtc.ICECandidate) {
|
|
if c != nil {
|
|
pc1.AddICECandidate(c.ToJSON())
|
|
}
|
|
})
|
|
|
|
d1, err := pc1.CreateDataChannel("loopback", nil)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
ch1 := make(chan error, 1)
|
|
d1.OnOpen(func() {
|
|
err := d1.Send([]byte(time.Now().Format(time.RFC3339Nano)))
|
|
if err != nil {
|
|
select {
|
|
case ch1 <- err:
|
|
default:
|
|
}
|
|
}
|
|
})
|
|
|
|
offer, err := pc1.CreateOffer(nil)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
err = pc1.SetLocalDescription(offer)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
err = pc2.SetRemoteDescription(*pc1.LocalDescription())
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
answer, err := pc2.CreateAnswer(nil)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
err = pc2.SetLocalDescription(answer)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
err = pc1.SetRemoteDescription(*pc2.LocalDescription())
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
ch2 := make(chan string, 1)
|
|
pc2.OnDataChannel(func(d2 *webrtc.DataChannel) {
|
|
d2.OnMessage(func(msg webrtc.DataChannelMessage) {
|
|
select {
|
|
case ch2 <- string(msg.Data):
|
|
default:
|
|
}
|
|
})
|
|
})
|
|
|
|
timer := time.NewTimer(timeout)
|
|
defer timer.Stop()
|
|
select {
|
|
case err := <-ch1:
|
|
return 0, err
|
|
case msg := <-ch2:
|
|
tm, err := time.Parse(time.RFC3339Nano, msg)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return time.Now().Sub(tm), nil
|
|
case <-timer.C:
|
|
return 0, errTimeout
|
|
}
|
|
}
|