mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-10 03:31:20 +00:00
71c8c75bc3
* Checkout shared/queue changes from hf1 branch * Remove logic that removes and reinserts an item into the queue at the back * Add test to ensure the correct thing was returned Co-authored-by: terence tsao <terence@prysmaticlabs.com>
270 lines
6.9 KiB
Go
270 lines
6.9 KiB
Go
package queue
|
|
|
|
import (
|
|
"container/heap"
|
|
"fmt"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/testutil/require"
|
|
)
|
|
|
|
// Ensure we satisfy the heap.Interface
|
|
var _ heap.Interface = &queue{}
|
|
|
|
// some tests rely on the ordering of items from this method
|
|
func testCases() (tc []*Item) {
|
|
// create a slice of items with priority / times offest by these seconds
|
|
for i, m := range []time.Duration{
|
|
5,
|
|
183600, // 51 hours
|
|
15, // 15 seconds
|
|
45, // 45 seconds
|
|
900, // 15 minutes
|
|
300, // 5 minutes
|
|
7200, // 2 hours
|
|
183600, // 51 hours
|
|
7201, // 2 hours, 1 second
|
|
115200, // 32 hours
|
|
1209600, // 2 weeks
|
|
} {
|
|
n := time.Now()
|
|
ft := n.Add(time.Second * m)
|
|
tc = append(tc, &Item{
|
|
Key: fmt.Sprintf("item-%d", i),
|
|
Value: 1,
|
|
Priority: ft.Unix(),
|
|
})
|
|
}
|
|
return
|
|
}
|
|
|
|
func TestPriorityQueue_New(t *testing.T) {
|
|
pq := New()
|
|
|
|
if len(pq.data) != len(pq.dataMap) || len(pq.data) != 0 {
|
|
t.Fatalf("error in queue/map size, expected data and map to be initialized, got (%d) and (%d)", len(pq.data), len(pq.dataMap))
|
|
}
|
|
|
|
if pq.Len() != 0 {
|
|
t.Fatalf("expected new queue to have zero size, got (%d)", pq.Len())
|
|
}
|
|
}
|
|
|
|
func TestPriorityQueue_Push(t *testing.T) {
|
|
pq := New()
|
|
|
|
// don't allow nil pushing
|
|
if err := pq.Push(nil); err == nil {
|
|
t.Fatal("Expected error on pushing nil")
|
|
}
|
|
|
|
tc := testCases()
|
|
tcl := len(tc)
|
|
for _, i := range tc {
|
|
if err := pq.Push(i); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
if pq.Len() != tcl {
|
|
t.Fatalf("error adding items, expected (%d) items, got (%d)", tcl, pq.Len())
|
|
}
|
|
|
|
testValidateInternalData(t, pq, len(tc), false)
|
|
|
|
item, err := pq.Pop()
|
|
if err != nil {
|
|
t.Fatalf("error popping item: %s", err)
|
|
}
|
|
if tc[0].Priority != item.Priority {
|
|
t.Fatalf("expected tc[0] and popped item to match, got (%q) and (%q)", tc[0], item.Priority)
|
|
}
|
|
if tc[0].Key != item.Key {
|
|
t.Fatalf("expected tc[0] and popped item to match, got (%q) and (%q)", tc[0], item.Priority)
|
|
}
|
|
|
|
testValidateInternalData(t, pq, len(tc)-1, false)
|
|
|
|
// push item with no key
|
|
dErr := pq.Push(tc[1])
|
|
if dErr != ErrDuplicateItem {
|
|
t.Fatal(err)
|
|
}
|
|
// push item with no key
|
|
tc[2].Key = ""
|
|
kErr := pq.Push(tc[2])
|
|
if kErr != nil && kErr.Error() != "error adding item: Item Key is required" {
|
|
t.Fatal(kErr)
|
|
}
|
|
|
|
testValidateInternalData(t, pq, len(tc)-1, false)
|
|
|
|
// check nil,nil error for not found
|
|
i, err := pq.PopByKey("empty")
|
|
if err != nil && i != nil {
|
|
t.Fatalf("expected nil error for PopByKey of non-existing key, got: %s", err)
|
|
}
|
|
}
|
|
|
|
func TestPriorityQueue_Pop(t *testing.T) {
|
|
pq := New()
|
|
|
|
tc := testCases()
|
|
for _, i := range tc {
|
|
if err := pq.Push(i); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
topItem, err := pq.Pop()
|
|
if err != nil {
|
|
t.Fatalf("error calling pop: %s", err)
|
|
}
|
|
if tc[0].Priority != topItem.Priority {
|
|
t.Fatalf("expected tc[0] and popped item to match, got (%q) and (%q)", tc[0], topItem.Priority)
|
|
}
|
|
if tc[0].Key != topItem.Key {
|
|
t.Fatalf("expected tc[0] and popped item to match, got (%q) and (%q)", tc[0], topItem.Priority)
|
|
}
|
|
|
|
var items []*Item
|
|
items = append(items, topItem)
|
|
// pop the remaining items, compare size of input and output
|
|
i, err := pq.Pop()
|
|
require.NoError(t, err)
|
|
for ; i != nil; i, err = pq.Pop() {
|
|
require.NoError(t, err)
|
|
items = append(items, i)
|
|
}
|
|
require.Equal(t, len(tc), len(items))
|
|
}
|
|
|
|
func TestPriorityQueue_PopByKey(t *testing.T) {
|
|
pq := New()
|
|
|
|
tc := testCases()
|
|
for _, i := range tc {
|
|
if err := pq.Push(i); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
// grab the top priority item, to capture it's value for checking later
|
|
item, err := pq.Pop()
|
|
require.NoError(t, err)
|
|
oldPriority := item.Priority
|
|
oldKey := item.Key
|
|
|
|
// push the item back on, so it gets removed with PopByKey and we verify
|
|
// the top item has changed later
|
|
require.NoError(t, pq.Push(item))
|
|
|
|
popKeys := []int{2, 4, 7, 1, 0}
|
|
for _, i := range popKeys {
|
|
item, err := pq.PopByKey(fmt.Sprintf("item-%d", i))
|
|
if err != nil {
|
|
t.Fatalf("failed to pop item-%d, \n\terr: %s\n\titem: %#v", i, err, item)
|
|
}
|
|
}
|
|
|
|
testValidateInternalData(t, pq, len(tc)-len(popKeys), false)
|
|
|
|
// grab the top priority item again, to compare with the top item priority
|
|
// from above
|
|
item, err = pq.Pop()
|
|
require.NoError(t, err)
|
|
newPriority := item.Priority
|
|
newKey := item.Key
|
|
|
|
if oldPriority == newPriority || oldKey == newKey {
|
|
t.Fatalf("expected old/new key and priority to differ, got (%s/%s) and (%d/%d)", oldKey, newKey, oldPriority, newPriority)
|
|
}
|
|
|
|
testValidateInternalData(t, pq, len(tc)-len(popKeys)-1, true)
|
|
}
|
|
|
|
func TestPriorityQueue_RetrieveByKey(t *testing.T) {
|
|
pq := New()
|
|
|
|
tc := testCases()
|
|
for _, i := range tc {
|
|
if err := pq.Push(i); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
// grab the top priority item, to capture it's value for checking later
|
|
item, err := pq.Pop()
|
|
require.NoError(t, err)
|
|
oldPriority := item.Priority
|
|
oldKey := item.Key
|
|
|
|
// push the item back on, so it gets retrieved with RetrieveByKey and we verify
|
|
// the top item does not change.
|
|
require.NoError(t, pq.Push(item))
|
|
|
|
popKeys := []int{2, 4, 7, 1, 0}
|
|
for _, i := range popKeys {
|
|
item := pq.RetrieveByKey(fmt.Sprintf("item-%d", i))
|
|
if item == nil {
|
|
t.Fatalf("failed to pop item-%d, \n\titem: %#v", i, item)
|
|
}
|
|
}
|
|
|
|
testValidateInternalData(t, pq, len(tc), false)
|
|
|
|
// grab the top priority item again, to compare with the top item priority
|
|
// from above. They should be the same
|
|
item, err = pq.Pop()
|
|
require.NoError(t, err)
|
|
newPriority := item.Priority
|
|
newKey := item.Key
|
|
|
|
if oldPriority != newPriority && oldKey != newKey {
|
|
t.Fatalf("expected old/new key and priority to be the same, got (%s/%s) and (%d/%d)", oldKey, newKey, oldPriority, newPriority)
|
|
}
|
|
|
|
testValidateInternalData(t, pq, len(tc)-1, true)
|
|
|
|
// ensure the correct item is retrieved when multiple items exist
|
|
require.NoError(t, pq.Push(&Item{
|
|
Key: "baz",
|
|
Value: "foo",
|
|
}))
|
|
require.NoError(t, pq.Push(&Item{
|
|
Key: "foo",
|
|
Value: "bar",
|
|
}))
|
|
i := pq.RetrieveByKey("foo")
|
|
require.Equal(t, "bar", i.Value)
|
|
|
|
}
|
|
|
|
// testValidateInternalData checks the internal data structure of the PriorityQueue
|
|
// and verifies that items are in-sync. Use drain only at the end of a test,
|
|
// because it will mutate the input queue
|
|
func testValidateInternalData(t *testing.T, pq *PriorityQueue, expectedSize int, drain bool) {
|
|
actualSize := pq.Len()
|
|
if actualSize != expectedSize {
|
|
t.Fatalf("expected new queue size to be (%d), got (%d)", expectedSize, actualSize)
|
|
}
|
|
|
|
if len(pq.data) != len(pq.dataMap) || len(pq.data) != expectedSize {
|
|
t.Fatalf("error in queue/map size, expected data and map to be (%d), got (%d) and (%d)", expectedSize, len(pq.data), len(pq.dataMap))
|
|
}
|
|
|
|
if drain && pq.Len() > 0 {
|
|
// pop all the items, verify lengths
|
|
i, err := pq.Pop()
|
|
require.NoError(t, err)
|
|
for ; i != nil; i, err = pq.Pop() {
|
|
require.NoError(t, err)
|
|
expectedSize--
|
|
if len(pq.data) != len(pq.dataMap) || len(pq.data) != expectedSize {
|
|
t.Fatalf("error in queue/map size, expected data and map to be (%d), got (%d) and (%d)", expectedSize, len(pq.data), len(pq.dataMap))
|
|
}
|
|
}
|
|
}
|
|
}
|