2014-11-27 23:14:41 +00:00
|
|
|
// Copyright (C) 2014 The Syncthing Authors.
|
|
|
|
//
|
2015-03-07 21:36:35 +01:00
|
|
|
// 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/.
|
2014-11-27 23:14:41 +00:00
|
|
|
|
|
|
|
package discover
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"testing"
|
|
|
|
|
2015-01-13 13:22:56 +01:00
|
|
|
"github.com/syncthing/protocol"
|
2015-04-22 23:54:31 +01:00
|
|
|
|
2015-08-06 11:29:25 +02:00
|
|
|
"github.com/syncthing/syncthing/lib/sync"
|
2014-11-27 23:14:41 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var device protocol.DeviceID
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
device, _ = protocol.DeviceIDFromString("P56IOI7-MZJNU2Y-IQGDREY-DM2MGTI-MGL3BXN-PQ6W5BM-TBBZ4TJ-XZWICQ2")
|
|
|
|
}
|
|
|
|
|
2015-07-23 20:01:25 +01:00
|
|
|
type FakeAnnouncer struct {
|
|
|
|
pkt Announce
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *FakeAnnouncer) Announcement() Announce {
|
|
|
|
return f.pkt
|
|
|
|
}
|
|
|
|
|
2014-11-27 23:14:41 +00:00
|
|
|
func TestUDP4Success(t *testing.T) {
|
|
|
|
conn, err := net.ListenUDP("udp4", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
port := conn.LocalAddr().(*net.UDPAddr).Port
|
|
|
|
|
|
|
|
address := fmt.Sprintf("udp4://127.0.0.1:%d", port)
|
2015-07-23 20:01:25 +01:00
|
|
|
pkt := Announce{
|
2014-11-27 23:14:41 +00:00
|
|
|
Magic: AnnouncementMagic,
|
|
|
|
This: Device{
|
|
|
|
device[:],
|
2015-06-23 13:55:30 +01:00
|
|
|
[]string{"tcp://123.123.123.123:1234"},
|
|
|
|
nil,
|
2014-11-27 23:14:41 +00:00
|
|
|
},
|
|
|
|
}
|
2015-07-23 20:01:25 +01:00
|
|
|
ann := &FakeAnnouncer{
|
|
|
|
pkt: pkt,
|
|
|
|
}
|
2014-11-27 23:14:41 +00:00
|
|
|
|
2015-07-23 20:01:25 +01:00
|
|
|
client, err := New(address, ann)
|
2014-11-27 23:14:41 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
udpclient := client.(*UDPClient)
|
|
|
|
if udpclient.errorRetryInterval != DefaultErrorRetryInternval {
|
|
|
|
t.Fatal("Incorrect retry interval")
|
|
|
|
}
|
|
|
|
|
|
|
|
if udpclient.listenAddress.IP != nil || udpclient.listenAddress.Port != 0 {
|
|
|
|
t.Fatal("Wrong listen IP or port", udpclient.listenAddress)
|
|
|
|
}
|
|
|
|
|
|
|
|
if client.Address() != address {
|
|
|
|
t.Fatal("Incorrect address")
|
|
|
|
}
|
|
|
|
|
|
|
|
buf := make([]byte, 2048)
|
|
|
|
|
|
|
|
// First announcement
|
|
|
|
conn.SetDeadline(time.Now().Add(time.Millisecond * 100))
|
|
|
|
_, err = conn.Read(buf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Announcement verification
|
|
|
|
conn.SetDeadline(time.Now().Add(time.Millisecond * 1100))
|
|
|
|
_, addr, err := conn.ReadFromUDP(buf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reply to it.
|
|
|
|
_, err = conn.WriteToUDP(pkt.MustMarshalXDR(), addr)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should get nothing else
|
|
|
|
conn.SetDeadline(time.Now().Add(time.Millisecond * 100))
|
|
|
|
_, err = conn.Read(buf)
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal("Expected error")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Status should be ok
|
|
|
|
if !client.StatusOK() {
|
|
|
|
t.Fatal("Wrong status")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do a lookup in a separate routine
|
|
|
|
addrs := []string{}
|
2015-04-22 23:54:31 +01:00
|
|
|
wg := sync.NewWaitGroup()
|
2014-11-27 23:14:41 +00:00
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
2015-06-23 13:55:30 +01:00
|
|
|
pkt, err := client.Lookup(device)
|
|
|
|
if err == nil {
|
|
|
|
for _, addr := range pkt.This.Addresses {
|
|
|
|
addrs = append(addrs, addr)
|
|
|
|
}
|
|
|
|
}
|
2014-11-27 23:14:41 +00:00
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Receive the lookup and reply
|
|
|
|
conn.SetDeadline(time.Now().Add(time.Millisecond * 100))
|
|
|
|
_, addr, err = conn.ReadFromUDP(buf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
conn.WriteToUDP(pkt.MustMarshalXDR(), addr)
|
|
|
|
|
|
|
|
// Wait for the lookup to arrive, verify that the number of answers is correct
|
|
|
|
wg.Wait()
|
|
|
|
|
2015-06-23 13:55:30 +01:00
|
|
|
if len(addrs) != 1 || addrs[0] != "tcp://123.123.123.123:1234" {
|
2014-11-27 23:14:41 +00:00
|
|
|
t.Fatal("Wrong number of answers")
|
|
|
|
}
|
|
|
|
|
|
|
|
client.Stop()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestUDP4Failure(t *testing.T) {
|
|
|
|
conn, err := net.ListenUDP("udp4", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
port := conn.LocalAddr().(*net.UDPAddr).Port
|
|
|
|
|
|
|
|
address := fmt.Sprintf("udp4://127.0.0.1:%d/?listenaddress=127.0.0.1&retry=5", port)
|
|
|
|
|
2015-07-23 20:01:25 +01:00
|
|
|
pkt := Announce{
|
2014-11-27 23:14:41 +00:00
|
|
|
Magic: AnnouncementMagic,
|
|
|
|
This: Device{
|
|
|
|
device[:],
|
2015-06-23 13:55:30 +01:00
|
|
|
[]string{"tcp://123.123.123.123:1234"},
|
|
|
|
nil,
|
2014-11-27 23:14:41 +00:00
|
|
|
},
|
|
|
|
}
|
2015-07-23 20:01:25 +01:00
|
|
|
ann := &FakeAnnouncer{
|
|
|
|
pkt: pkt,
|
|
|
|
}
|
2014-11-27 23:14:41 +00:00
|
|
|
|
2015-07-23 20:01:25 +01:00
|
|
|
client, err := New(address, ann)
|
2014-11-27 23:14:41 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
udpclient := client.(*UDPClient)
|
|
|
|
if udpclient.errorRetryInterval != time.Second*5 {
|
|
|
|
t.Fatal("Incorrect retry interval")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !udpclient.listenAddress.IP.Equal(net.IPv4(127, 0, 0, 1)) || udpclient.listenAddress.Port != 0 {
|
|
|
|
t.Fatal("Wrong listen IP or port", udpclient.listenAddress)
|
|
|
|
}
|
|
|
|
|
|
|
|
if client.Address() != address {
|
|
|
|
t.Fatal("Incorrect address")
|
|
|
|
}
|
|
|
|
|
|
|
|
buf := make([]byte, 2048)
|
|
|
|
|
|
|
|
// First announcement
|
|
|
|
conn.SetDeadline(time.Now().Add(time.Millisecond * 100))
|
|
|
|
_, err = conn.Read(buf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Announcement verification
|
|
|
|
conn.SetDeadline(time.Now().Add(time.Millisecond * 1100))
|
|
|
|
_, _, err = conn.ReadFromUDP(buf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Don't reply
|
|
|
|
// We should get nothing else
|
|
|
|
conn.SetDeadline(time.Now().Add(time.Millisecond * 100))
|
|
|
|
_, err = conn.Read(buf)
|
|
|
|
if err == nil {
|
|
|
|
t.Fatal("Expected error")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Status should be failure
|
|
|
|
if client.StatusOK() {
|
|
|
|
t.Fatal("Wrong status")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do a lookup in a separate routine
|
|
|
|
addrs := []string{}
|
2015-04-22 23:54:31 +01:00
|
|
|
wg := sync.NewWaitGroup()
|
2014-11-27 23:14:41 +00:00
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
2015-06-23 13:55:30 +01:00
|
|
|
pkt, err := client.Lookup(device)
|
|
|
|
if err == nil {
|
|
|
|
for _, addr := range pkt.This.Addresses {
|
|
|
|
addrs = append(addrs, addr)
|
|
|
|
}
|
|
|
|
}
|
2014-11-27 23:14:41 +00:00
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Receive the lookup and don't reply
|
|
|
|
conn.SetDeadline(time.Now().Add(time.Millisecond * 100))
|
|
|
|
_, _, err = conn.ReadFromUDP(buf)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait for the lookup to timeout, verify that the number of answers is none
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
if len(addrs) != 0 {
|
|
|
|
t.Fatal("Wrong number of answers")
|
|
|
|
}
|
|
|
|
|
|
|
|
client.Stop()
|
|
|
|
}
|