forked from nictuku/dht
-
Notifications
You must be signed in to change notification settings - Fork 0
/
routing_test.go
208 lines (192 loc) · 5.33 KB
/
routing_test.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
package dht
import (
"crypto/rand"
"fmt"
"sort"
"strings"
"testing"
)
// 16 bytes.
const ffff = "\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff"
func BenchmarkInsertRecursive(b *testing.B) {
b.StopTimer()
// Add 1k nodes to the tree.
const count = 1000
nodes := make([]*remoteNode, 0, count)
for i := 0; i < count; i++ {
rId := make([]byte, 4)
if _, err := rand.Read(rId); err != nil {
b.Fatal("Couldnt produce random numbers for FindClosest:", err)
}
id := string(rId) + ffff
if len(id) != 20 {
b.Fatalf("Random infohash construction error, wrong len: want %d, got %d",
20, len(id))
}
nodes = append(nodes, &remoteNode{id: id})
}
b.StartTimer()
// Each op is adding 1000 nodes to the tree.
for i := 0; i < b.N; i++ {
tree := &nTree{}
for _, r := range nodes {
tree.insert(r)
}
}
}
func BenchmarkFindClosest(b *testing.B) {
b.StopTimer()
node, err := NewDHTNode(0, 1e7, false)
node.nodeId = "00bcdefghij01234567"
if err != nil {
b.Fatal(err)
}
// Add 100k nodes to the remote nodes slice.
for i := 0; i < 100000; i++ {
rId := make([]byte, 4)
if _, err := rand.Read(rId); err != nil {
b.Fatal("Couldnt produce random numbers for FindClosest:", err)
}
r := &remoteNode{
lastQueryID: 0,
id: string(rId) + ffff,
}
if len(r.id) != 20 {
b.Fatalf("remoteNode construction error, wrong len: want %d, got %d",
20, len(r.id))
}
r.reachable = true
node.routingTable.insert(r)
}
b.StartTimer()
for i := 0; i < b.N; i++ {
f := node.routingTable.lookupFiltered(InfoHash(fmt.Sprintf("x%10v", i) + "xxxxxxxxx"))
if len(f) != kNodes {
b.Fatalf("Missing results. Wanted %d, got %d", kNodes, len(f))
}
}
}
type testData struct {
query string
want int // just the size.
}
var nodes = []*remoteNode{
{id: "\x00"},
{id: "\x01"},
{id: "\x02"},
{id: "\x03"},
{id: "\x04"},
{id: "\x05"},
{id: "\x06"},
{id: "\x07"},
{id: "\x08"},
{id: "\x09"},
{id: "\x10"},
}
func TestNodeDelete(t *testing.T) {
tree := &nTree{}
for _, r := range nodes[:4] {
tree.insert(r)
}
for i, r := range []string{"\x00", "\x01"} {
id := InfoHash(r)
t.Logf("Removing node: %x", r)
tree.cut(id, 0)
neighbors := tree.lookup(id)
if len(neighbors) == 0 {
t.Errorf("Deleted too many nodes.")
}
if len(neighbors) != 3-i {
t.Errorf("Too many nodes left in the tree: got %d, wanted %d", len(neighbors), 3-i)
}
if r == neighbors[0].id {
t.Errorf("Node didnt get deleted as expected: %x", r)
}
}
}
func TestNodeDistance(t *testing.T) {
tree := &nTree{}
for _, r := range nodes {
r.reachable = true
tree.insert(r)
}
tests := []testData{
{"\x04", 8},
{"\x07", 8},
}
for _, r := range tests {
q := InfoHash(r.query)
distances := make([]string, 0, len(tests))
neighbors := tree.lookup(q)
if len(neighbors) != r.want {
t.Errorf("id: %x, wanted len=%d, got len=%d", q, r.want, len(neighbors))
t.Errorf("Details: %#v", neighbors)
}
for _, x := range neighbors {
d := hashDistance(q, InfoHash(x.id))
var b []string
for _, c := range d {
if c != 0 {
b = append(b, fmt.Sprintf("%08b", c))
} else {
b = append(b, "00000000")
}
}
d = strings.Join(b, ".")
distances = append(distances, d)
}
if !sort.StringsAreSorted(distances) {
t.Errorf("Resulting distances for %x are not sorted", r.query)
for i, d := range distances {
t.Errorf("id: %x, d: %v", neighbors[i].id, d)
}
}
}
}
// ===================== lookup benchmark =================================
// $ go test -v -bench='BenchmarkFindClosest' -run=NONE
//
// #1 In hindsight, this was a very embarrasing first attempt. I kept a list of
// my nodes, and every time I had to do a lookup, I re-sorted the whole list of
// nodes in the routing table using the XOR distance to the target infohash.
// Honestly I had no idea how bad this was when I was wrote it. :-)
// BenchmarkFindClosest 1 7020661000 ns/op
//
// #2 not-checked in attempt to use a trie. Not even correct.
// BenchmarkFindClosest 1 1072682000 ns/op
//
// #3 only compare bytes that we need.
// BenchmarkFindClosest 1 1116333000 ns/op
//
// #4 moved to buckets, but using only one.
// BenchmarkFindClosest 1 1170809000 ns/op
//
// #5 using my new nTree (not yet correct)
// BenchmarkFindClosest 100000 27194 ns/op
//
// #6 recursive nTree (correct)
// BenchmarkFindClosest 200000 10585 ns/op
//
// #7 removed an unnecessary wrapper function.
// BenchmarkFindClosest 200000 9691 ns/op
//
// #8 Random infohashes now have identical suffix instead of prefix. In the
// wild, most of the calculations are done in the most significant bits so this
// is closer to reality.
// BenchmarkFindClosest 50000 35165 ns/op
//
// #9 Suffix compression. Magic? :-)
// BenchmarkFindClosest 1000000 2795 ns/op
// ===================== insertion benchmark =================================
// $ go test -v -bench='BenchmarkInsert.*' -run=none
//
// #1 initial version of the test.
// BenchmarkInsertRecursive 500 4701600 ns/op
// BenchmarkInsert 500 3595448 ns/op
//
// #2 Random infohashes have identical suffix instead of prefix.
// BenchmarkInsertRecursive 100 22598150 ns/op
// BenchmarkInsert 100 19239120 ns/op
//
// #3 Suffix compression. Much less work (iterative version removed).
// BenchmarkInsertRecursive 5000 448471 ns/op