mirror of
https://github.com/status-im/status-go.git
synced 2025-01-09 14:16:21 +00:00
58fcf809ea
* 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
79 lines
1.5 KiB
Go
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)
|
|
}
|