mirror of
https://github.com/syncthing/syncthing.git
synced 2024-11-17 19:08:46 -07:00
2a886576a6
In the successfull case, we start the timer with NewTimer(0), then do a bunch of stuff during which time it can fire, then reset it with Reset(0). The result is that two timer firings are queued when we enter the select loop, so we do two announcements back to back and fail the tests.
247 lines
5.5 KiB
Go
247 lines
5.5 KiB
Go
// Copyright (C) 2014 The Syncthing Authors.
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify it
|
|
// under the terms of the GNU General Public License as published by the Free
|
|
// Software Foundation, either version 3 of the License, or (at your option)
|
|
// any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful, but WITHOUT
|
|
// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
// more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License along
|
|
// with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package discover
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"io"
|
|
"net"
|
|
"net/url"
|
|
"strconv"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/syncthing/syncthing/internal/protocol"
|
|
)
|
|
|
|
func init() {
|
|
for _, proto := range []string{"udp", "udp4", "udp6"} {
|
|
Register(proto, func(uri *url.URL, pkt *Announce) (Client, error) {
|
|
c := &UDPClient{}
|
|
err := c.Start(uri, pkt)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return c, nil
|
|
})
|
|
}
|
|
}
|
|
|
|
type UDPClient struct {
|
|
url *url.URL
|
|
|
|
id protocol.DeviceID
|
|
|
|
stop chan struct{}
|
|
wg sync.WaitGroup
|
|
listenAddress *net.UDPAddr
|
|
|
|
globalBroadcastInterval time.Duration
|
|
errorRetryInterval time.Duration
|
|
|
|
status bool
|
|
mut sync.RWMutex
|
|
}
|
|
|
|
func (d *UDPClient) Start(uri *url.URL, pkt *Announce) error {
|
|
d.url = uri
|
|
d.id = protocol.DeviceIDFromBytes(pkt.This.ID)
|
|
d.stop = make(chan struct{})
|
|
|
|
params := uri.Query()
|
|
// The address must not have a port, as otherwise both announce and lookup
|
|
// sockets would try to bind to the same port.
|
|
addr, err := net.ResolveUDPAddr(d.url.Scheme, params.Get("listenaddress")+":0")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
d.listenAddress = addr
|
|
|
|
broadcastSeconds, err := strconv.ParseUint(params.Get("broadcast"), 0, 0)
|
|
if err != nil {
|
|
d.globalBroadcastInterval = DefaultGlobalBroadcastInterval
|
|
} else {
|
|
d.globalBroadcastInterval = time.Duration(broadcastSeconds) * time.Second
|
|
}
|
|
|
|
retrySeconds, err := strconv.ParseUint(params.Get("retry"), 0, 0)
|
|
if err != nil {
|
|
d.errorRetryInterval = DefaultErrorRetryInternval
|
|
} else {
|
|
d.errorRetryInterval = time.Duration(retrySeconds) * time.Second
|
|
}
|
|
|
|
d.wg.Add(1)
|
|
go d.broadcast(pkt.MustMarshalXDR())
|
|
return nil
|
|
}
|
|
|
|
func (d *UDPClient) broadcast(pkt []byte) {
|
|
defer d.wg.Done()
|
|
|
|
conn, err := net.ListenUDP(d.url.Scheme, d.listenAddress)
|
|
for err != nil {
|
|
l.Warnf("Global UDP discovery (%s): %v; trying again in %v", d.url, err, d.errorRetryInterval)
|
|
select {
|
|
case <-d.stop:
|
|
return
|
|
case <-time.After(d.errorRetryInterval):
|
|
}
|
|
conn, err = net.ListenUDP(d.url.Scheme, d.listenAddress)
|
|
}
|
|
defer conn.Close()
|
|
|
|
remote, err := net.ResolveUDPAddr(d.url.Scheme, d.url.Host)
|
|
for err != nil {
|
|
l.Warnf("Global UDP discovery (%s): %v; trying again in %v", d.url, err, d.errorRetryInterval)
|
|
select {
|
|
case <-d.stop:
|
|
return
|
|
case <-time.After(d.errorRetryInterval):
|
|
}
|
|
remote, err = net.ResolveUDPAddr(d.url.Scheme, d.url.Host)
|
|
}
|
|
|
|
timer := time.NewTimer(0)
|
|
for {
|
|
select {
|
|
case <-d.stop:
|
|
return
|
|
|
|
case <-timer.C:
|
|
var ok bool
|
|
|
|
if debug {
|
|
l.Debugf("Global UDP discovery (%s): send announcement -> %v\n%s", d.url, remote, hex.Dump(pkt))
|
|
}
|
|
|
|
_, err := conn.WriteTo(pkt, remote)
|
|
if err != nil {
|
|
if debug {
|
|
l.Debugf("discover %s: warning: %s", d.url, err)
|
|
}
|
|
ok = false
|
|
} else {
|
|
// Verify that the announce server responds positively for our device ID
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
res := d.Lookup(d.id)
|
|
if debug {
|
|
l.Debugf("discover %s: external lookup check: %v", d.url, res)
|
|
}
|
|
ok = len(res) > 0
|
|
}
|
|
|
|
d.mut.Lock()
|
|
d.status = ok
|
|
d.mut.Unlock()
|
|
|
|
if ok {
|
|
timer.Reset(d.globalBroadcastInterval)
|
|
} else {
|
|
timer.Reset(d.errorRetryInterval)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (d *UDPClient) Lookup(device protocol.DeviceID) []string {
|
|
extIP, err := net.ResolveUDPAddr(d.url.Scheme, d.url.Host)
|
|
if err != nil {
|
|
if debug {
|
|
l.Debugf("discover %s: %v; no external lookup", d.url, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
conn, err := net.DialUDP(d.url.Scheme, d.listenAddress, extIP)
|
|
if err != nil {
|
|
if debug {
|
|
l.Debugf("discover %s: %v; no external lookup", d.url, err)
|
|
}
|
|
return nil
|
|
}
|
|
defer conn.Close()
|
|
|
|
err = conn.SetDeadline(time.Now().Add(5 * time.Second))
|
|
if err != nil {
|
|
if debug {
|
|
l.Debugf("discover %s: %v; no external lookup", d.url, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
buf := Query{QueryMagic, device[:]}.MustMarshalXDR()
|
|
_, err = conn.Write(buf)
|
|
if err != nil {
|
|
if debug {
|
|
l.Debugf("discover %s: %v; no external lookup", d.url, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
buf = make([]byte, 2048)
|
|
n, err := conn.Read(buf)
|
|
if err != nil {
|
|
if err, ok := err.(net.Error); ok && err.Timeout() {
|
|
// Expected if the server doesn't know about requested device ID
|
|
return nil
|
|
}
|
|
if debug {
|
|
l.Debugf("discover %s: %v; no external lookup", d.url, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
if debug {
|
|
l.Debugf("discover %s: read external:\n%s", d.url, hex.Dump(buf[:n]))
|
|
}
|
|
|
|
var pkt Announce
|
|
err = pkt.UnmarshalXDR(buf[:n])
|
|
if err != nil && err != io.EOF {
|
|
if debug {
|
|
l.Debugln("discover %s:", d.url, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
var addrs []string
|
|
for _, a := range pkt.This.Addresses {
|
|
deviceAddr := net.JoinHostPort(net.IP(a.IP).String(), strconv.Itoa(int(a.Port)))
|
|
addrs = append(addrs, deviceAddr)
|
|
}
|
|
return addrs
|
|
}
|
|
|
|
func (d *UDPClient) Stop() {
|
|
if d.stop != nil {
|
|
close(d.stop)
|
|
d.wg.Wait()
|
|
}
|
|
}
|
|
|
|
func (d *UDPClient) StatusOK() bool {
|
|
d.mut.RLock()
|
|
defer d.mut.RUnlock()
|
|
return d.status
|
|
}
|
|
|
|
func (d *UDPClient) Address() string {
|
|
return d.url.String()
|
|
}
|