2016-05-04 19:38:12 +00:00
// Copyright (C) 2015 The Syncthing Authors.
//
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
// You can obtain one at http://mozilla.org/MPL/2.0/.
package connections
import (
"crypto/tls"
2016-05-17 00:05:38 +00:00
"errors"
2016-05-04 19:38:12 +00:00
"fmt"
"io"
"net"
"net/url"
"time"
"github.com/juju/ratelimit"
"github.com/syncthing/syncthing/lib/config"
"github.com/syncthing/syncthing/lib/discover"
"github.com/syncthing/syncthing/lib/events"
"github.com/syncthing/syncthing/lib/nat"
"github.com/syncthing/syncthing/lib/protocol"
"github.com/syncthing/syncthing/lib/sync"
"github.com/syncthing/syncthing/lib/util"
// Registers NAT service providers
_ "github.com/syncthing/syncthing/lib/pmp"
_ "github.com/syncthing/syncthing/lib/upnp"
"github.com/thejerf/suture"
)
var (
dialers = make ( map [ string ] dialerFactory , 0 )
listeners = make ( map [ string ] listenerFactory , 0 )
)
2016-07-02 20:33:31 +00:00
const (
perDeviceWarningRate = 1.0 / ( 15 * 60 ) // Once per 15 minutes
tlsHandshakeTimeout = 10 * time . Second
)
2016-06-09 12:30:35 +00:00
2016-05-04 19:38:12 +00:00
// Service listens and dials all configured unconnected devices, via supported
// dialers. Successful connections are handed to the model.
type Service struct {
* suture . Supervisor
cfg * config . Wrapper
myID protocol . DeviceID
model Model
tlsCfg * tls . Config
discoverer discover . Finder
2016-11-30 07:54:20 +00:00
conns chan internalConn
2016-05-04 19:38:12 +00:00
bepProtocolName string
tlsDefaultCommonName string
lans [ ] * net . IPNet
writeRateLimit * ratelimit . Bucket
readRateLimit * ratelimit . Bucket
natService * nat . Service
natServiceToken * suture . ServiceToken
2016-11-19 12:37:14 +00:00
listenersMut sync . RWMutex
listeners map [ string ] genericListener
listenerTokens map [ string ] suture . ServiceToken
listenerSupervisor * suture . Supervisor
2016-05-09 15:03:12 +00:00
curConMut sync . Mutex
2016-11-30 07:54:20 +00:00
currentConnection map [ protocol . DeviceID ] completeConn
2016-05-04 19:38:12 +00:00
}
func NewService ( cfg * config . Wrapper , myID protocol . DeviceID , mdl Model , tlsCfg * tls . Config , discoverer discover . Finder ,
bepProtocolName string , tlsDefaultCommonName string , lans [ ] * net . IPNet ) * Service {
service := & Service {
2016-11-19 12:37:14 +00:00
Supervisor : suture . New ( "connections.Service" , suture . Spec {
Log : func ( line string ) {
l . Infoln ( line )
} ,
} ) ,
2016-05-04 19:38:12 +00:00
cfg : cfg ,
myID : myID ,
model : mdl ,
tlsCfg : tlsCfg ,
discoverer : discoverer ,
2016-11-30 07:54:20 +00:00
conns : make ( chan internalConn ) ,
2016-05-04 19:38:12 +00:00
bepProtocolName : bepProtocolName ,
tlsDefaultCommonName : tlsDefaultCommonName ,
lans : lans ,
natService : nat . NewService ( myID , cfg ) ,
2016-05-09 15:03:12 +00:00
listenersMut : sync . NewRWMutex ( ) ,
listeners : make ( map [ string ] genericListener ) ,
listenerTokens : make ( map [ string ] suture . ServiceToken ) ,
2016-11-19 12:37:14 +00:00
// A listener can fail twice, rapidly. Any more than that and it
// will be put on suspension for ten minutes. Restarts and changes
// due to config are done by removing and adding services, so are
// not subject to these limitations.
listenerSupervisor : suture . New ( "c.S.listenerSupervisor" , suture . Spec {
Log : func ( line string ) {
l . Infoln ( line )
} ,
FailureThreshold : 2 ,
FailureBackoff : 600 * time . Second ,
} ) ,
2016-05-09 15:03:12 +00:00
curConMut : sync . NewMutex ( ) ,
2016-11-30 07:54:20 +00:00
currentConnection : make ( map [ protocol . DeviceID ] completeConn ) ,
2016-05-04 19:38:12 +00:00
}
cfg . Subscribe ( service )
// The rate variables are in KiB/s in the UI (despite the camel casing
// of the name). We multiply by 1024 here to get B/s.
2016-05-24 21:57:56 +00:00
options := service . cfg . Options ( )
if options . MaxSendKbps > 0 {
service . writeRateLimit = ratelimit . NewBucketWithRate ( float64 ( 1024 * options . MaxSendKbps ) , int64 ( 5 * 1024 * options . MaxSendKbps ) )
2016-05-04 19:38:12 +00:00
}
2016-05-24 21:57:56 +00:00
if options . MaxRecvKbps > 0 {
service . readRateLimit = ratelimit . NewBucketWithRate ( float64 ( 1024 * options . MaxRecvKbps ) , int64 ( 5 * 1024 * options . MaxRecvKbps ) )
2016-05-04 19:38:12 +00:00
}
// There are several moving parts here; one routine per listening address
// (handled in configuration changing) to handle incoming connections,
// one routine to periodically attempt outgoing connections, one routine to
// the the common handling regardless of whether the connection was
// incoming or outgoing.
service . Add ( serviceFunc ( service . connect ) )
service . Add ( serviceFunc ( service . handle ) )
2016-11-19 12:37:14 +00:00
service . Add ( service . listenerSupervisor )
2016-05-04 19:38:12 +00:00
2016-11-12 09:34:18 +00:00
raw := cfg . RawCopy ( )
2016-05-04 19:38:12 +00:00
// Actually starts the listeners and NAT service
service . CommitConfiguration ( raw , raw )
return service
}
2016-05-17 00:05:38 +00:00
var (
errDisabled = errors . New ( "disabled by configuration" )
)
2016-05-04 19:38:12 +00:00
func ( s * Service ) handle ( ) {
next :
for c := range s . conns {
cs := c . ConnectionState ( )
// We should have negotiated the next level protocol "bep/1.0" as part
// of the TLS handshake. Unfortunately this can't be a hard error,
// because there are implementations out there that don't support
// protocol negotiation (iOS for one...).
if ! cs . NegotiatedProtocolIsMutual || cs . NegotiatedProtocol != s . bepProtocolName {
l . Infof ( "Peer %s did not negotiate bep/1.0" , c . RemoteAddr ( ) )
}
// We should have received exactly one certificate from the other
// side. If we didn't, they don't have a device ID and we drop the
// connection.
certs := cs . PeerCertificates
if cl := len ( certs ) ; cl != 1 {
l . Infof ( "Got peer certificate list of length %d != 1 from %s; protocol error" , cl , c . RemoteAddr ( ) )
c . Close ( )
continue
}
remoteCert := certs [ 0 ]
remoteID := protocol . NewDeviceID ( remoteCert . Raw )
// The device ID should not be that of ourselves. It can happen
// though, especially in the presence of NAT hairpinning, multiple
// clients between the same NAT gateway, and global discovery.
if remoteID == s . myID {
l . Infof ( "Connected to myself (%s) - should not happen" , remoteID )
c . Close ( )
continue
}
2016-06-09 10:50:14 +00:00
c . SetDeadline ( time . Now ( ) . Add ( 20 * time . Second ) )
hello , err := protocol . ExchangeHello ( c , s . model . GetHello ( remoteID ) )
2016-05-04 19:38:12 +00:00
if err != nil {
2016-06-09 10:50:14 +00:00
if protocol . IsVersionMismatch ( err ) {
// The error will be a relatively user friendly description
2016-07-04 10:40:29 +00:00
// of what's wrong with the version compatibility. By
// default identify the other side by device ID and IP.
remote := fmt . Sprintf ( "%v (%v)" , remoteID , c . RemoteAddr ( ) )
if hello . DeviceName != "" {
// If the name was set in the hello return, use that to
// give the user more info about which device is the
// affected one. It probably says more than the remote
// IP.
remote = fmt . Sprintf ( "%q (%s %s, %v)" , hello . DeviceName , hello . ClientName , hello . ClientVersion , remoteID )
}
msg := fmt . Sprintf ( "Connecting to %s: %s" , remote , err )
2016-06-09 12:30:35 +00:00
warningFor ( remoteID , msg )
2016-06-09 10:50:14 +00:00
} else {
// It's something else - connection reset or whatever
l . Infof ( "Failed to exchange Hello messages with %s (%s): %s" , remoteID , c . RemoteAddr ( ) , err )
}
2016-05-04 19:38:12 +00:00
c . Close ( )
continue
}
2016-06-09 10:50:14 +00:00
c . SetDeadline ( time . Time { } )
2016-05-04 19:38:12 +00:00
2016-08-05 09:29:49 +00:00
// The Model will return an error for devices that we don't want to
// have a connection with for whatever reason, for example unknown devices.
if err := s . model . OnHello ( remoteID , c . RemoteAddr ( ) , hello ) ; err != nil {
2016-12-02 11:56:14 +00:00
l . Infof ( "Connection from %s at %s (%s) rejected: %v" , remoteID , c . RemoteAddr ( ) , c . Type ( ) , err )
2016-08-05 09:29:49 +00:00
c . Close ( )
continue
}
2016-05-04 19:38:12 +00:00
// If we have a relay connection, and the new incoming connection is
// not a relay connection, we should drop that, and prefer the this one.
2016-08-07 12:20:37 +00:00
connected := s . model . ConnectedTo ( remoteID )
2016-05-09 15:03:12 +00:00
s . curConMut . Lock ( )
2016-05-04 19:38:12 +00:00
ct , ok := s . currentConnection [ remoteID ]
2016-05-09 15:03:12 +00:00
s . curConMut . Unlock ( )
2016-08-07 12:20:37 +00:00
priorityKnown := ok && connected
2016-05-04 19:38:12 +00:00
// Lower priority is better, just like nice etc.
2016-11-30 07:54:20 +00:00
if priorityKnown && ct . internalConn . priority > c . priority {
2016-05-04 19:38:12 +00:00
l . Debugln ( "Switching connections" , remoteID )
2016-08-07 12:20:37 +00:00
} else if connected {
2016-05-04 19:38:12 +00:00
// We should not already be connected to the other party. TODO: This
// could use some better handling. If the old connection is dead but
// hasn't timed out yet we may want to drop *that* connection and keep
// this one. But in case we are two devices connecting to each other
// in parallel we don't want to do that or we end up with no
// connections still established...
l . Infof ( "Connected to already connected device (%s)" , remoteID )
c . Close ( )
2016-05-09 15:03:12 +00:00
continue
2016-05-04 19:38:12 +00:00
}
2016-08-05 09:29:49 +00:00
deviceCfg , ok := s . cfg . Device ( remoteID )
if ! ok {
panic ( "bug: unknown device should already have been rejected" )
}
2016-05-04 19:38:12 +00:00
2016-08-05 09:29:49 +00:00
// Verify the name on the certificate. By default we set it to
// "syncthing" when generating, but the user may have replaced
// the certificate and used another name.
certName := deviceCfg . CertName
if certName == "" {
certName = s . tlsDefaultCommonName
}
if err := remoteCert . VerifyHostname ( certName ) ; err != nil {
// Incorrect certificate name is something the user most
// likely wants to know about, since it's an advanced
// config. Warn instead of Info.
l . Warnf ( "Bad certificate from %s (%v): %v" , remoteID , c . RemoteAddr ( ) , err )
c . Close ( )
continue next
}
2016-05-04 19:38:12 +00:00
2016-08-05 09:29:49 +00:00
// If rate limiting is set, and based on the address we should
// limit the connection, then we wrap it in a limiter.
2016-05-04 19:38:12 +00:00
2016-08-05 09:29:49 +00:00
limit := s . shouldLimit ( c . RemoteAddr ( ) )
2016-05-04 19:38:12 +00:00
2016-08-05 09:29:49 +00:00
wr := io . Writer ( c )
if limit && s . writeRateLimit != nil {
wr = NewWriteLimiter ( c , s . writeRateLimit )
}
2016-05-04 19:38:12 +00:00
2016-08-05 09:29:49 +00:00
rd := io . Reader ( c )
if limit && s . readRateLimit != nil {
rd = NewReadLimiter ( c , s . readRateLimit )
}
2016-05-04 19:38:12 +00:00
2016-11-30 07:54:20 +00:00
name := fmt . Sprintf ( "%s-%s (%s)" , c . LocalAddr ( ) , c . RemoteAddr ( ) , c . Type ( ) )
2016-08-05 09:29:49 +00:00
protoConn := protocol . NewConnection ( remoteID , rd , wr , s . model , name , deviceCfg . Compression )
2016-11-30 07:54:20 +00:00
modelConn := completeConn { c , protoConn }
2016-05-04 19:38:12 +00:00
2016-08-05 09:29:49 +00:00
l . Infof ( "Established secure connection to %s at %s" , remoteID , name )
l . Debugf ( "cipher suite: %04X in lan: %t" , c . ConnectionState ( ) . CipherSuite , ! limit )
2016-05-04 19:38:12 +00:00
2016-08-05 09:29:49 +00:00
s . model . AddConnection ( modelConn , hello )
s . curConMut . Lock ( )
s . currentConnection [ remoteID ] = modelConn
s . curConMut . Unlock ( )
continue next
2016-05-04 19:38:12 +00:00
}
}
func ( s * Service ) connect ( ) {
nextDial := make ( map [ string ] time . Time )
2016-05-09 15:03:12 +00:00
2016-05-17 00:05:38 +00:00
// Used as delay for the first few connection attempts, increases
// exponentially
initialRampup := time . Second
// Calculated from actual dialers reconnectInterval
var sleep time . Duration
2016-05-09 15:33:25 +00:00
2016-05-04 19:38:12 +00:00
for {
2016-11-12 09:34:18 +00:00
cfg := s . cfg . RawCopy ( )
2016-05-17 00:05:38 +00:00
bestDialerPrio := 1 << 31 - 1 // worse prio won't build on 32 bit
for _ , df := range dialers {
if ! df . Enabled ( cfg ) {
continue
}
if prio := df . Priority ( ) ; prio < bestDialerPrio {
bestDialerPrio = prio
}
}
2016-05-04 19:38:12 +00:00
l . Debugln ( "Reconnect loop" )
now := time . Now ( )
var seen [ ] string
nextDevice :
2016-05-17 00:05:38 +00:00
for _ , deviceCfg := range cfg . Devices {
deviceID := deviceCfg . DeviceID
2016-05-04 19:38:12 +00:00
if deviceID == s . myID {
continue
}
paused := s . model . IsPaused ( deviceID )
if paused {
continue
}
2016-05-09 15:03:12 +00:00
connected := s . model . ConnectedTo ( deviceID )
s . curConMut . Lock ( )
2016-08-07 12:20:37 +00:00
ct , ok := s . currentConnection [ deviceID ]
2016-05-09 15:03:12 +00:00
s . curConMut . Unlock ( )
2016-08-07 12:20:37 +00:00
priorityKnown := ok && connected
2016-05-09 15:03:12 +00:00
2016-11-30 07:54:20 +00:00
if priorityKnown && ct . internalConn . priority == bestDialerPrio {
2016-05-09 15:33:25 +00:00
// Things are already as good as they can get.
continue
}
l . Debugln ( "Reconnect loop for" , deviceID )
2016-05-04 19:38:12 +00:00
var addrs [ ] string
for _ , addr := range deviceCfg . Addresses {
if addr == "dynamic" {
if s . discoverer != nil {
if t , err := s . discoverer . Lookup ( deviceID ) ; err == nil {
addrs = append ( addrs , t ... )
}
}
} else {
addrs = append ( addrs , addr )
}
}
seen = append ( seen , addrs ... )
for _ , addr := range addrs {
2016-05-17 00:05:38 +00:00
nextDialAt , ok := nextDial [ addr ]
if ok && initialRampup >= sleep && nextDialAt . After ( now ) {
l . Debugf ( "Not dialing %v as sleep is %v, next dial is at %s and current time is %s" , addr , sleep , nextDialAt , now )
continue
}
// If we fail at any step before actually getting the dialer
// retry in a minute
nextDial [ addr ] = now . Add ( time . Minute )
2016-05-04 19:38:12 +00:00
uri , err := url . Parse ( addr )
if err != nil {
2016-05-17 00:05:38 +00:00
l . Infof ( "Dialer for %s: %v" , addr , err )
2016-05-04 19:38:12 +00:00
continue
}
2016-05-17 00:05:38 +00:00
dialerFactory , err := s . getDialerFactory ( cfg , uri )
if err == errDisabled {
l . Debugln ( "Dialer for" , uri , "is disabled" )
2016-05-04 19:38:12 +00:00
continue
}
2016-05-17 00:05:38 +00:00
if err != nil {
l . Infof ( "Dialer for %v: %v" , uri , err )
2016-05-04 19:38:12 +00:00
continue
}
2016-11-30 07:54:20 +00:00
if priorityKnown && dialerFactory . Priority ( ) >= ct . internalConn . priority {
l . Debugf ( "Not dialing using %s as priority is less than current connection (%d >= %d)" , dialerFactory , dialerFactory . Priority ( ) , ct . internalConn . priority )
2016-05-04 19:38:12 +00:00
continue
}
2016-05-17 00:05:38 +00:00
dialer := dialerFactory . New ( s . cfg , s . tlsCfg )
2016-05-04 19:38:12 +00:00
l . Debugln ( "dial" , deviceCfg . DeviceID , uri )
2016-05-17 00:05:38 +00:00
nextDial [ addr ] = now . Add ( dialer . RedialFrequency ( ) )
2016-05-04 19:38:12 +00:00
conn , err := dialer . Dial ( deviceID , uri )
if err != nil {
l . Debugln ( "dial failed" , deviceCfg . DeviceID , uri , err )
continue
}
s . conns <- conn
continue nextDevice
}
}
nextDial , sleep = filterAndFindSleepDuration ( nextDial , seen , now )
2016-05-17 00:05:38 +00:00
if initialRampup < sleep {
l . Debugln ( "initial rampup; sleep" , initialRampup , "and update to" , initialRampup * 2 )
time . Sleep ( initialRampup )
initialRampup *= 2
2016-05-04 19:38:12 +00:00
} else {
2016-05-17 00:05:38 +00:00
l . Debugln ( "sleep until next dial" , sleep )
2016-05-04 19:38:12 +00:00
time . Sleep ( sleep )
}
}
}
func ( s * Service ) shouldLimit ( addr net . Addr ) bool {
if s . cfg . Options ( ) . LimitBandwidthInLan {
return true
}
tcpaddr , ok := addr . ( * net . TCPAddr )
if ! ok {
return true
}
for _ , lan := range s . lans {
if lan . Contains ( tcpaddr . IP ) {
return false
}
}
return ! tcpaddr . IP . IsLoopback ( )
}
2016-05-17 00:05:38 +00:00
func ( s * Service ) createListener ( factory listenerFactory , uri * url . URL ) bool {
2016-05-09 15:03:12 +00:00
// must be called with listenerMut held
2016-05-04 19:38:12 +00:00
2016-05-17 00:05:38 +00:00
l . Debugln ( "Starting listener" , uri )
2016-05-04 19:38:12 +00:00
2016-05-17 00:05:38 +00:00
listener := factory . New ( uri , s . cfg , s . tlsCfg , s . conns , s . natService )
2016-05-04 19:38:12 +00:00
listener . OnAddressesChanged ( s . logListenAddressesChangedEvent )
2016-05-17 00:05:38 +00:00
s . listeners [ uri . String ( ) ] = listener
2016-11-19 12:37:14 +00:00
s . listenerTokens [ uri . String ( ) ] = s . listenerSupervisor . Add ( listener )
2016-05-17 00:05:38 +00:00
return true
2016-05-04 19:38:12 +00:00
}
func ( s * Service ) logListenAddressesChangedEvent ( l genericListener ) {
events . Default . Log ( events . ListenAddressesChanged , map [ string ] interface { } {
"address" : l . URI ( ) ,
"lan" : l . LANAddresses ( ) ,
"wan" : l . WANAddresses ( ) ,
} )
}
func ( s * Service ) VerifyConfiguration ( from , to config . Configuration ) error {
return nil
}
func ( s * Service ) CommitConfiguration ( from , to config . Configuration ) bool {
newDevices := make ( map [ protocol . DeviceID ] bool , len ( to . Devices ) )
for _ , dev := range to . Devices {
newDevices [ dev . DeviceID ] = true
}
for _ , dev := range from . Devices {
if ! newDevices [ dev . DeviceID ] {
2016-08-07 16:21:59 +00:00
s . curConMut . Lock ( )
delete ( s . currentConnection , dev . DeviceID )
s . curConMut . Unlock ( )
warningLimitersMut . Lock ( )
delete ( warningLimiters , dev . DeviceID )
warningLimitersMut . Unlock ( )
2016-05-04 19:38:12 +00:00
}
}
2016-05-09 15:03:12 +00:00
s . listenersMut . Lock ( )
2016-05-04 19:38:12 +00:00
seen := make ( map [ string ] struct { } )
for _ , addr := range config . Wrap ( "" , to ) . ListenAddresses ( ) {
2016-05-17 00:05:38 +00:00
if _ , ok := s . listeners [ addr ] ; ok {
seen [ addr ] = struct { } { }
continue
2016-05-04 19:38:12 +00:00
}
2016-05-17 00:05:38 +00:00
uri , err := url . Parse ( addr )
if err != nil {
l . Infof ( "Listener for %s: %v" , addr , err )
continue
}
factory , err := s . getListenerFactory ( to , uri )
if err == errDisabled {
l . Debugln ( "Listener for" , uri , "is disabled" )
continue
}
if err != nil {
l . Infof ( "Listener for %v: %v" , uri , err )
continue
}
s . createListener ( factory , uri )
2016-05-04 19:38:12 +00:00
seen [ addr ] = struct { } { }
}
2016-05-17 00:05:38 +00:00
for addr , listener := range s . listeners {
if _ , ok := seen [ addr ] ; ! ok || ! listener . Factory ( ) . Enabled ( to ) {
2016-05-04 19:38:12 +00:00
l . Debugln ( "Stopping listener" , addr )
2016-11-19 12:37:14 +00:00
s . listenerSupervisor . Remove ( s . listenerTokens [ addr ] )
2016-05-04 19:38:12 +00:00
delete ( s . listenerTokens , addr )
delete ( s . listeners , addr )
}
}
2016-05-09 15:03:12 +00:00
s . listenersMut . Unlock ( )
2016-05-04 19:38:12 +00:00
if to . Options . NATEnabled && s . natServiceToken == nil {
l . Debugln ( "Starting NAT service" )
token := s . Add ( s . natService )
s . natServiceToken = & token
} else if ! to . Options . NATEnabled && s . natServiceToken != nil {
l . Debugln ( "Stopping NAT service" )
s . Remove ( * s . natServiceToken )
s . natServiceToken = nil
}
2016-08-07 16:21:59 +00:00
return true
2016-05-04 19:38:12 +00:00
}
func ( s * Service ) AllAddresses ( ) [ ] string {
2016-05-09 15:03:12 +00:00
s . listenersMut . RLock ( )
2016-05-04 19:38:12 +00:00
var addrs [ ] string
for _ , listener := range s . listeners {
for _ , lanAddr := range listener . LANAddresses ( ) {
addrs = append ( addrs , lanAddr . String ( ) )
}
for _ , wanAddr := range listener . WANAddresses ( ) {
addrs = append ( addrs , wanAddr . String ( ) )
}
}
2016-05-09 15:03:12 +00:00
s . listenersMut . RUnlock ( )
2016-05-04 19:38:12 +00:00
return util . UniqueStrings ( addrs )
}
func ( s * Service ) ExternalAddresses ( ) [ ] string {
2016-05-09 15:03:12 +00:00
s . listenersMut . RLock ( )
2016-05-04 19:38:12 +00:00
var addrs [ ] string
for _ , listener := range s . listeners {
for _ , wanAddr := range listener . WANAddresses ( ) {
addrs = append ( addrs , wanAddr . String ( ) )
}
}
2016-05-09 15:03:12 +00:00
s . listenersMut . RUnlock ( )
2016-05-04 19:38:12 +00:00
return util . UniqueStrings ( addrs )
}
func ( s * Service ) Status ( ) map [ string ] interface { } {
2016-05-09 15:03:12 +00:00
s . listenersMut . RLock ( )
2016-05-04 19:38:12 +00:00
result := make ( map [ string ] interface { } )
for addr , listener := range s . listeners {
status := make ( map [ string ] interface { } )
err := listener . Error ( )
if err != nil {
status [ "error" ] = err . Error ( )
}
status [ "lanAddresses" ] = urlsToStrings ( listener . LANAddresses ( ) )
status [ "wanAddresses" ] = urlsToStrings ( listener . WANAddresses ( ) )
result [ addr ] = status
}
2016-05-09 15:03:12 +00:00
s . listenersMut . RUnlock ( )
2016-05-04 19:38:12 +00:00
return result
}
2016-05-17 00:05:38 +00:00
func ( s * Service ) getDialerFactory ( cfg config . Configuration , uri * url . URL ) ( dialerFactory , error ) {
dialerFactory , ok := dialers [ uri . Scheme ]
if ! ok {
return nil , fmt . Errorf ( "unknown address scheme %q" , uri . Scheme )
}
if ! dialerFactory . Enabled ( cfg ) {
return nil , errDisabled
}
return dialerFactory , nil
}
func ( s * Service ) getListenerFactory ( cfg config . Configuration , uri * url . URL ) ( listenerFactory , error ) {
listenerFactory , ok := listeners [ uri . Scheme ]
if ! ok {
return nil , fmt . Errorf ( "unknown address scheme %q" , uri . Scheme )
}
if ! listenerFactory . Enabled ( cfg ) {
return nil , errDisabled
}
return listenerFactory , nil
}
2016-05-04 19:38:12 +00:00
func filterAndFindSleepDuration ( nextDial map [ string ] time . Time , seen [ ] string , now time . Time ) ( map [ string ] time . Time , time . Duration ) {
newNextDial := make ( map [ string ] time . Time )
for _ , addr := range seen {
nextDialAt , ok := nextDial [ addr ]
if ok {
newNextDial [ addr ] = nextDialAt
}
}
min := time . Minute
for _ , next := range newNextDial {
cur := next . Sub ( now )
if cur < min {
min = cur
}
}
return newNextDial , min
}
func urlsToStrings ( urls [ ] * url . URL ) [ ] string {
strings := make ( [ ] string , len ( urls ) )
for i , url := range urls {
strings [ i ] = url . String ( )
}
return strings
}
2016-06-09 12:30:35 +00:00
var warningLimiters = make ( map [ protocol . DeviceID ] * ratelimit . Bucket )
var warningLimitersMut = sync . NewMutex ( )
func warningFor ( dev protocol . DeviceID , msg string ) {
warningLimitersMut . Lock ( )
defer warningLimitersMut . Unlock ( )
lim , ok := warningLimiters [ dev ]
if ! ok {
lim = ratelimit . NewBucketWithRate ( perDeviceWarningRate , 1 )
warningLimiters [ dev ] = lim
}
if lim . TakeAvailable ( 1 ) == 1 {
l . Warnln ( msg )
}
}
2016-07-02 20:33:31 +00:00
func tlsTimedHandshake ( tc * tls . Conn ) error {
tc . SetDeadline ( time . Now ( ) . Add ( tlsHandshakeTimeout ) )
defer tc . SetDeadline ( time . Time { } )
return tc . Handshake ( )
}