status-go/peers/topic_peer_queue_test.go
Andrea Maria Piana 58fcf809ea
Bug/fix pending peers segfault (#2004)
* Replace mclock with time in peer pools

we used mclock as golang before 1.9 did not support monotonic clocks,
https://github.com/gavv/monotime, it does now https://golang.org/pkg/time/
so we can fallback on the system implementation, which will return
nanoseconds with a resolution that is system dependent.

* Handle case where peer have same discovered time

In some system the resolution of the clock is not high enough so
multiple peers are added on the same nanosecond.

This result in the peer just added being immediately removed.

This code adds a check making sure we don't assume that a peer is added.

Another approach would be to make sure to include the peer in the list,
so prevent the peer just being added to be evicted, but it's slightly
more complicated and the resolution is generally accurate enough for our
purpose so that peers will be fresh enough if they have the same
discovered time.

It also adds a regression test, I had to use an interface to stub the
clock.

Fixes: https://github.com/status-im/nim-status-client/issues/522

* bump version to 0.55.3
2020-07-24 11:14:05 -04:00

79 lines
1.5 KiB
Go

package peers
import (
"container/heap"
"math/rand"
"testing"
"time"
"github.com/stretchr/testify/require"
)
func TestPeerPriorityQueueSorting(t *testing.T) {
count := 5
discTimes := make([]time.Time, count)
// generate a slice of monotonic times
for i := 0; i < count; i++ {
discTimes[i] = time.Now()
}
// shuffle discTimes
for i := range discTimes {
j := rand.Intn(i + 1)
discTimes[i], discTimes[j] = discTimes[j], discTimes[i]
}
// make a priority queue
q := make(peerPriorityQueue, count)
for i := 0; i < count; i++ {
q[i] = &peerInfoItem{
peerInfo: &peerInfo{
discoveredTime: discTimes[i],
},
}
}
heap.Init(&q)
// verify that the slice is sorted ascending by `discoveredTime`
var item *peerInfoItem
for q.Len() > 0 {
newItem := heap.Pop(&q).(*peerInfoItem)
if item != nil {
require.True(t, item.discoveredTime.After(newItem.discoveredTime))
}
item = newItem
}
}
func TestPeerPriorityQueueIndexUpdating(t *testing.T) {
q := make(peerPriorityQueue, 0)
heap.Init(&q)
item1 := &peerInfoItem{
index: -1,
peerInfo: &peerInfo{
discoveredTime: time.Now(),
},
}
item2 := &peerInfoItem{
index: -1,
peerInfo: &peerInfo{
discoveredTime: time.Now(),
},
}
// insert older item first
heap.Push(&q, item1)
require.Equal(t, item1.index, 0)
heap.Push(&q, item2)
require.Equal(t, item2.index, 0)
require.Equal(t, item1.index, 1)
// pop operation should reset index
poppedItem := heap.Pop(&q)
require.Equal(t, item2, poppedItem)
require.Equal(t, item2.index, -1)
require.Equal(t, item1.index, 0)
}