2015-09-30 21:38:59 +02: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,
|
2017-02-09 07:52:18 +01:00
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
2015-09-30 21:38:59 +02:00
|
|
|
|
2015-09-29 17:40:29 +02:00
|
|
|
package discover
|
|
|
|
|
|
|
|
import (
|
2020-02-13 14:43:00 +01:00
|
|
|
"context"
|
2020-08-18 08:26:33 +01:00
|
|
|
"crypto/tls"
|
2015-09-29 17:40:29 +02:00
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2020-08-18 08:26:33 +01:00
|
|
|
"github.com/syncthing/syncthing/lib/config"
|
|
|
|
"github.com/syncthing/syncthing/lib/events"
|
2015-09-29 17:40:29 +02:00
|
|
|
"github.com/syncthing/syncthing/lib/protocol"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestCacheUnique(t *testing.T) {
|
2017-11-22 07:05:49 +00:00
|
|
|
addresses0 := []string{"tcp://192.0.2.44:22000", "tcp://192.0.2.42:22000"}
|
|
|
|
addresses1 := []string{"tcp://192.0.2.43:22000", "tcp://192.0.2.42:22000"}
|
2015-09-30 22:24:06 +02:00
|
|
|
|
2016-05-04 19:38:12 +00:00
|
|
|
// what we expect from just addresses0
|
|
|
|
addresses0Sorted := []string{"tcp://192.0.2.42:22000", "tcp://192.0.2.44:22000"}
|
2015-09-30 22:24:06 +02:00
|
|
|
|
2016-05-04 19:38:12 +00:00
|
|
|
// what we expect from addresses0+addresses1
|
2015-09-30 22:24:06 +02:00
|
|
|
totalSorted := []string{
|
2017-11-22 07:05:49 +00:00
|
|
|
"tcp://192.0.2.42:22000",
|
2015-09-30 22:24:06 +02:00
|
|
|
// no duplicate .42
|
2017-11-22 07:05:49 +00:00
|
|
|
"tcp://192.0.2.43:22000",
|
|
|
|
"tcp://192.0.2.44:22000",
|
2015-09-30 22:24:06 +02:00
|
|
|
}
|
|
|
|
|
2020-08-18 08:26:33 +01:00
|
|
|
cfg := config.New(protocol.LocalDeviceID)
|
|
|
|
cfg.Options.LocalAnnEnabled = false
|
|
|
|
cfg.Options.GlobalAnnEnabled = false
|
|
|
|
|
|
|
|
c := NewManager(protocol.LocalDeviceID, config.Wrap("", cfg, events.NoopLogger), tls.Certificate{}, events.NoopLogger, nil).(*manager)
|
|
|
|
c.ServeBackground()
|
2015-09-29 17:40:29 +02:00
|
|
|
defer c.Stop()
|
|
|
|
|
2017-11-04 07:20:11 +00:00
|
|
|
// Add a fake discovery service and verify we get its answers through the
|
2015-09-29 17:40:29 +02:00
|
|
|
// cache.
|
|
|
|
|
2016-05-04 19:38:12 +00:00
|
|
|
f1 := &fakeDiscovery{addresses0}
|
2020-08-18 08:26:33 +01:00
|
|
|
c.addLocked("f1", f1, time.Minute, 0)
|
2015-09-29 17:40:29 +02:00
|
|
|
|
2020-02-13 14:43:00 +01:00
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
addr, err := c.Lookup(ctx, protocol.LocalDeviceID)
|
2015-09-29 17:40:29 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2016-05-04 19:38:12 +00:00
|
|
|
if !reflect.DeepEqual(addr, addresses0Sorted) {
|
|
|
|
t.Errorf("Incorrect addresses; %+v != %+v", addr, addresses0Sorted)
|
2015-09-29 17:40:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add one more that answers in the same way and check that we don't
|
|
|
|
// duplicate or otherwise mess up the responses now.
|
|
|
|
|
2016-05-04 19:38:12 +00:00
|
|
|
f2 := &fakeDiscovery{addresses1}
|
2020-08-18 08:26:33 +01:00
|
|
|
c.addLocked("f2", f2, time.Minute, 0)
|
2015-09-29 17:40:29 +02:00
|
|
|
|
2020-02-13 14:43:00 +01:00
|
|
|
addr, err = c.Lookup(ctx, protocol.LocalDeviceID)
|
2015-09-29 17:40:29 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2016-05-04 19:38:12 +00:00
|
|
|
if !reflect.DeepEqual(addr, totalSorted) {
|
|
|
|
t.Errorf("Incorrect addresses; %+v != %+v", addr, totalSorted)
|
2015-09-29 17:40:29 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type fakeDiscovery struct {
|
2016-05-04 19:38:12 +00:00
|
|
|
addresses []string
|
2015-09-29 17:40:29 +02:00
|
|
|
}
|
|
|
|
|
2020-02-13 14:43:00 +01:00
|
|
|
func (f *fakeDiscovery) Lookup(_ context.Context, deviceID protocol.DeviceID) (addresses []string, err error) {
|
2016-05-04 19:38:12 +00:00
|
|
|
return f.addresses, nil
|
2015-09-29 17:40:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f *fakeDiscovery) Error() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *fakeDiscovery) String() string {
|
|
|
|
return "fake"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *fakeDiscovery) Cache() map[protocol.DeviceID]CacheEntry {
|
|
|
|
return nil
|
|
|
|
}
|
2016-01-09 00:28:13 +01:00
|
|
|
|
|
|
|
func TestCacheSlowLookup(t *testing.T) {
|
2020-08-18 08:26:33 +01:00
|
|
|
cfg := config.New(protocol.LocalDeviceID)
|
|
|
|
cfg.Options.LocalAnnEnabled = false
|
|
|
|
cfg.Options.GlobalAnnEnabled = false
|
|
|
|
|
|
|
|
c := NewManager(protocol.LocalDeviceID, config.Wrap("", cfg, events.NoopLogger), tls.Certificate{}, events.NoopLogger, nil).(*manager)
|
|
|
|
c.ServeBackground()
|
2016-01-09 00:28:13 +01:00
|
|
|
defer c.Stop()
|
|
|
|
|
|
|
|
// Add a slow discovery service.
|
|
|
|
|
|
|
|
started := make(chan struct{})
|
|
|
|
f1 := &slowDiscovery{time.Second, started}
|
2020-08-18 08:26:33 +01:00
|
|
|
c.addLocked("f1", f1, time.Minute, 0)
|
2016-01-09 00:28:13 +01:00
|
|
|
|
|
|
|
// Start a lookup, which will take at least a second
|
|
|
|
|
|
|
|
t0 := time.Now()
|
2020-02-13 14:43:00 +01:00
|
|
|
go c.Lookup(context.Background(), protocol.LocalDeviceID)
|
2016-01-09 00:28:13 +01:00
|
|
|
<-started // The slow lookup method has been called so we're inside the lock
|
|
|
|
|
|
|
|
// It should be possible to get ChildErrors while it's running
|
|
|
|
|
|
|
|
c.ChildErrors()
|
|
|
|
|
|
|
|
// Only a small amount of time should have passed, not the full second
|
|
|
|
|
|
|
|
diff := time.Since(t0)
|
|
|
|
if diff > 500*time.Millisecond {
|
|
|
|
t.Error("ChildErrors was blocked for", diff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type slowDiscovery struct {
|
|
|
|
delay time.Duration
|
|
|
|
started chan struct{}
|
|
|
|
}
|
|
|
|
|
2020-02-13 14:43:00 +01:00
|
|
|
func (f *slowDiscovery) Lookup(_ context.Context, deviceID protocol.DeviceID) (addresses []string, err error) {
|
2016-01-09 00:28:13 +01:00
|
|
|
close(f.started)
|
|
|
|
time.Sleep(f.delay)
|
2016-05-04 19:38:12 +00:00
|
|
|
return nil, nil
|
2016-01-09 00:28:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (f *slowDiscovery) Error() error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *slowDiscovery) String() string {
|
|
|
|
return "fake"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *slowDiscovery) Cache() map[protocol.DeviceID]CacheEntry {
|
|
|
|
return nil
|
|
|
|
}
|