Merge pull request #2448 from calmh/connhandling

Refactor out methods resolveAddresses, connectDirect, connectViaRelay
This commit is contained in:
Audrius Butkevicius 2015-11-09 22:01:06 -05:00
commit 48ce356d5c

View File

@ -280,50 +280,19 @@ func (s *connectionSvc) connect() {
continue continue
} }
var addrs []string addrs, relays := s.resolveAddresses(deviceID, deviceCfg.Addresses)
var relays []discover.Relay
for _, addr := range deviceCfg.Addresses {
if addr == "dynamic" {
if s.discoverer != nil {
if t, r, err := s.discoverer.Lookup(deviceID); err == nil {
addrs = append(addrs, t...)
relays = append(relays, r...)
}
}
} else {
addrs = append(addrs, addr)
}
}
for _, addr := range addrs { for _, addr := range addrs {
uri, err := url.Parse(addr) if conn := s.connectDirect(deviceID, addr); conn != nil {
if err != nil {
l.Infoln("Failed to parse connection url:", addr, err)
continue
}
dialer, ok := dialers[uri.Scheme]
if !ok {
l.Infoln("Unknown address schema", uri)
continue
}
l.Debugln("dial", deviceCfg.DeviceID, uri)
conn, err := dialer(uri, s.tlsCfg)
if err != nil {
l.Debugln("dial failed", deviceCfg.DeviceID, uri, err)
continue
}
if connected { if connected {
s.model.Close(deviceID, fmt.Errorf("switching connections")) s.model.Close(deviceID, fmt.Errorf("switching connections"))
} }
s.conns <- model.IntermediateConnection{ s.conns <- model.IntermediateConnection{
conn, model.ConnectionTypeDirectDial, conn, model.ConnectionTypeDirectDial,
} }
continue nextDevice continue nextDevice
} }
}
// Only connect via relays if not already connected // Only connect via relays if not already connected
// Also, do not set lastRelayCheck time if we have no relays, // Also, do not set lastRelayCheck time if we have no relays,
@ -345,47 +314,14 @@ func (s *connectionSvc) connect() {
s.lastRelayCheck[deviceID] = time.Now() s.lastRelayCheck[deviceID] = time.Now()
for _, addr := range relays { for _, addr := range relays {
uri, err := url.Parse(addr.URL) if conn := s.connectViaRelay(deviceID, addr); conn != nil {
if err != nil {
l.Infoln("Failed to parse relay connection url:", addr, err)
continue
}
inv, err := client.GetInvitationFromRelay(uri, deviceID, s.tlsCfg.Certificates)
if err != nil {
l.Debugf("Failed to get invitation for %s from %s: %v", deviceID, uri, err)
continue
} else {
l.Debugln("Succesfully retrieved relay invitation", inv, "from", uri)
}
conn, err := client.JoinSession(inv)
if err != nil {
l.Debugf("Failed to join relay session %s: %v", inv, err)
continue
} else {
l.Debugln("Sucessfully joined relay session", inv)
}
var tc *tls.Conn
if inv.ServerSocket {
tc = tls.Server(conn, s.tlsCfg)
} else {
tc = tls.Client(conn, s.tlsCfg)
}
err = tc.Handshake()
if err != nil {
l.Infof("TLS handshake (BEP/relay %s): %v", inv, err)
tc.Close()
continue
}
s.conns <- model.IntermediateConnection{ s.conns <- model.IntermediateConnection{
tc, model.ConnectionTypeRelayDial, conn, model.ConnectionTypeRelayDial,
} }
continue nextDevice continue nextDevice
} }
} }
}
time.Sleep(delay) time.Sleep(delay)
delay *= 2 delay *= 2
@ -395,6 +331,84 @@ func (s *connectionSvc) connect() {
} }
} }
func (s *connectionSvc) resolveAddresses(deviceID protocol.DeviceID, inAddrs []string) (addrs []string, relays []discover.Relay) {
for _, addr := range inAddrs {
if addr == "dynamic" {
if s.discoverer != nil {
if t, r, err := s.discoverer.Lookup(deviceID); err == nil {
addrs = append(addrs, t...)
relays = append(relays, r...)
}
}
} else {
addrs = append(addrs, addr)
}
}
return
}
func (s *connectionSvc) connectDirect(deviceID protocol.DeviceID, addr string) *tls.Conn {
uri, err := url.Parse(addr)
if err != nil {
l.Infoln("Failed to parse connection url:", addr, err)
return nil
}
dialer, ok := dialers[uri.Scheme]
if !ok {
l.Infoln("Unknown address schema", uri)
return nil
}
l.Debugln("dial", deviceID, uri)
conn, err := dialer(uri, s.tlsCfg)
if err != nil {
l.Debugln("dial failed", deviceID, uri, err)
return nil
}
return conn
}
func (s *connectionSvc) connectViaRelay(deviceID protocol.DeviceID, addr discover.Relay) *tls.Conn {
uri, err := url.Parse(addr.URL)
if err != nil {
l.Infoln("Failed to parse relay connection url:", addr, err)
return nil
}
inv, err := client.GetInvitationFromRelay(uri, deviceID, s.tlsCfg.Certificates)
if err != nil {
l.Debugf("Failed to get invitation for %s from %s: %v", deviceID, uri, err)
return nil
}
l.Debugln("Succesfully retrieved relay invitation", inv, "from", uri)
conn, err := client.JoinSession(inv)
if err != nil {
l.Debugf("Failed to join relay session %s: %v", inv, err)
return nil
}
l.Debugln("Sucessfully joined relay session", inv)
var tc *tls.Conn
if inv.ServerSocket {
tc = tls.Server(conn, s.tlsCfg)
} else {
tc = tls.Client(conn, s.tlsCfg)
}
err = tc.Handshake()
if err != nil {
l.Infof("TLS handshake (BEP/relay %s): %v", inv, err)
tc.Close()
return nil
}
return tc
}
func (s *connectionSvc) acceptRelayConns() { func (s *connectionSvc) acceptRelayConns() {
for { for {
conn := s.relaySvc.Accept() conn := s.relaySvc.Accept()