p2p/discover: deflake UDP tests

This commit is contained in:
Felix Lange 2015-02-09 11:02:32 +01:00
parent 028775a086
commit 9915d3c3be
4 changed files with 129 additions and 65 deletions

View File

@ -87,9 +87,10 @@ func (tab *Table) Lookup(target NodeID) []*Node {
reply = make(chan []*Node, alpha) reply = make(chan []*Node, alpha)
pendingQueries = 0 pendingQueries = 0
) )
// don't query further if we hit the target. // don't query further if we hit the target or ourself.
// unlikely to happen often in practice. // unlikely to happen often in practice.
asked[target] = true asked[target] = true
asked[tab.self.ID] = true
tab.mutex.Lock() tab.mutex.Lock()
// update last lookup stamp (for refresh logic) // update last lookup stamp (for refresh logic)

View File

@ -14,6 +14,18 @@ import (
"github.com/ethereum/go-ethereum/crypto" "github.com/ethereum/go-ethereum/crypto"
) )
func TestTable_bumpOrAddBucketAssign(t *testing.T) {
tab := newTable(nil, NodeID{}, &net.UDPAddr{})
for i := 1; i < len(tab.buckets); i++ {
tab.bumpOrAdd(randomID(tab.self.ID, i), &net.UDPAddr{})
}
for i, b := range tab.buckets {
if i > 0 && len(b.entries) != 1 {
t.Errorf("bucket %d has %d entries, want 1", i, len(b.entries))
}
}
}
func TestTable_bumpOrAddPingReplace(t *testing.T) { func TestTable_bumpOrAddPingReplace(t *testing.T) {
pingC := make(pingC) pingC := make(pingC)
tab := newTable(pingC, NodeID{}, &net.UDPAddr{}) tab := newTable(pingC, NodeID{}, &net.UDPAddr{})

View File

@ -179,10 +179,9 @@ func (t *udp) findnode(to *Node, target NodeID) ([]*Node, error) {
nreceived := 0 nreceived := 0
errc := t.pending(to.ID, neighborsPacket, func(r interface{}) bool { errc := t.pending(to.ID, neighborsPacket, func(r interface{}) bool {
reply := r.(*neighbors) reply := r.(*neighbors)
for i := 0; i < len(reply.Nodes); i++ { for _, n := range reply.Nodes {
nreceived++ nreceived++
n := reply.Nodes[i] if n.isValid() {
if n.ID != t.self.ID && n.isValid() {
nodes = append(nodes, n) nodes = append(nodes, n)
} }
} }

View File

@ -1,10 +1,10 @@
package discover package discover
import ( import (
"fmt"
logpkg "log" logpkg "log"
"net" "net"
"os" "os"
"reflect"
"testing" "testing"
"time" "time"
@ -53,16 +53,37 @@ func TestUDP_findnode(t *testing.T) {
defer n1.Close() defer n1.Close()
defer n2.Close() defer n2.Close()
entry := MustParseNode("enode://9d8a19597e312ef32d76e6b4903bb43d7bcd892d17b769d30b404bd3a4c2dca6c86184b17d0fdeeafe3b01e0e912d990ddc853db3f325d5419f31446543c30be@127.0.0.1:54194") // put a few nodes into n2. the exact distribution shouldn't
n2.add([]*Node{entry}) // matter much, altough we need to take care not to overflow
// any bucket.
target := randomID(n1.self.ID, 100) target := randomID(n1.self.ID, 100)
result, _ := n1.net.findnode(n2.self, target) nodes := &nodesByDistance{target: target}
if len(result) != 1 { for i := 0; i < bucketSize; i++ {
t.Fatalf("wrong number of results: got %d, want 1", len(result)) n2.add([]*Node{&Node{
IP: net.IP{1, 2, 3, byte(i)},
DiscPort: i + 2,
TCPPort: i + 2,
ID: randomID(n2.self.ID, i+2),
}})
} }
if !reflect.DeepEqual(result[0], entry) { n2.add(nodes.entries)
t.Errorf("wrong result: got %v, want %v", result[0], entry) n2.bumpOrAdd(n1.self.ID, &net.UDPAddr{IP: n1.self.IP, Port: n1.self.DiscPort})
expected := n2.closest(target, bucketSize)
err := runUDP(10, func() error {
result, _ := n1.net.findnode(n2.self, target)
if len(result) != bucketSize {
return fmt.Errorf("wrong number of results: got %d, want %d", len(result), bucketSize)
}
for i := range result {
if result[i].ID != expected.entries[i].ID {
return fmt.Errorf("result mismatch at %d:\n got: %v\n want: %v", i, result[i], expected.entries[i])
}
}
return nil
})
if err != nil {
t.Error(err)
} }
} }
@ -101,59 +122,90 @@ func TestUDP_findnodeMultiReply(t *testing.T) {
defer n1.Close() defer n1.Close()
defer n2.Close() defer n2.Close()
nodes := make([]*Node, bucketSize) err := runUDP(10, func() error {
for i := range nodes { nodes := make([]*Node, bucketSize)
nodes[i] = &Node{ for i := range nodes {
IP: net.IP{1, 2, 3, 4}, nodes[i] = &Node{
DiscPort: i + 1, IP: net.IP{1, 2, 3, 4},
TCPPort: i + 1, DiscPort: i + 1,
ID: randomID(n2.self.ID, i+1), TCPPort: i + 1,
} ID: randomID(n2.self.ID, i+1),
}
// ask N2 for neighbors. it will send an empty reply back.
// the request will wait for up to bucketSize replies.
resultC := make(chan []*Node)
go func() {
ns, err := n1.net.findnode(n2.self, n1.self.ID)
if err != nil {
t.Error("findnode error:", err)
}
resultC <- ns
}()
// send a few more neighbors packets to N1.
// it should collect those.
for end := 0; end < len(nodes); {
off := end
if end = end + 5; end > len(nodes) {
end = len(nodes)
}
udp2.send(n1.self, neighborsPacket, neighbors{
Nodes: nodes[off:end],
Expiration: uint64(time.Now().Add(10 * time.Second).Unix()),
})
}
// check that they are all returned. we cannot just check for
// equality because they might not be returned in the order they
// were sent.
result := <-resultC
if hasDuplicates(result) {
t.Error("result slice contains duplicates")
}
if len(result) != len(nodes) {
t.Errorf("wrong number of nodes returned: got %d, want %d", len(result), len(nodes))
}
matched := make(map[NodeID]bool)
for _, n := range result {
for _, expn := range nodes {
if n.ID == expn.ID { // && bytes.Equal(n.Addr.IP, expn.Addr.IP) && n.Addr.Port == expn.Addr.Port {
matched[n.ID] = true
} }
} }
}
if len(matched) != len(nodes) { // ask N2 for neighbors. it will send an empty reply back.
t.Errorf("wrong number of matching nodes: got %d, want %d", len(matched), len(nodes)) // the request will wait for up to bucketSize replies.
resultc := make(chan []*Node)
errc := make(chan error)
go func() {
ns, err := n1.net.findnode(n2.self, n1.self.ID)
if err != nil {
errc <- err
} else {
resultc <- ns
}
}()
// send a few more neighbors packets to N1.
// it should collect those.
for end := 0; end < len(nodes); {
off := end
if end = end + 5; end > len(nodes) {
end = len(nodes)
}
udp2.send(n1.self, neighborsPacket, neighbors{
Nodes: nodes[off:end],
Expiration: uint64(time.Now().Add(10 * time.Second).Unix()),
})
}
// check that they are all returned. we cannot just check for
// equality because they might not be returned in the order they
// were sent.
var result []*Node
select {
case result = <-resultc:
case err := <-errc:
return err
}
if hasDuplicates(result) {
return fmt.Errorf("result slice contains duplicates")
}
if len(result) != len(nodes) {
return fmt.Errorf("wrong number of nodes returned: got %d, want %d", len(result), len(nodes))
}
matched := make(map[NodeID]bool)
for _, n := range result {
for _, expn := range nodes {
if n.ID == expn.ID { // && bytes.Equal(n.Addr.IP, expn.Addr.IP) && n.Addr.Port == expn.Addr.Port {
matched[n.ID] = true
}
}
}
if len(matched) != len(nodes) {
return fmt.Errorf("wrong number of matching nodes: got %d, want %d", len(matched), len(nodes))
}
return nil
})
if err != nil {
t.Error(err)
} }
} }
// runUDP runs a test n times and returns an error if the test failed
// in all n runs. This is necessary because UDP is unreliable even for
// connections on the local machine, causing test failures.
func runUDP(n int, test func() error) error {
errcount := 0
errors := ""
for i := 0; i < n; i++ {
if err := test(); err != nil {
errors += fmt.Sprintf("\n#%d: %v", i, err)
errcount++
}
}
if errcount == n {
return fmt.Errorf("failed on all %d iterations:%s", n, errors)
}
return nil
}