package skiplist import ( "bytes" "fmt" "math/rand" "strconv" "testing" ) const ( maxN = 10000 ) var ( memStore = newMemStore() ) func TestInsertAndFind(t *testing.T) { k0 := []byte("0") var list *SkipList var listPointer *SkipList listPointer.Insert(k0, k0) if _, ok, _ := listPointer.Find(k0); ok { t.Fail() } list = New(memStore) if _, ok, _ := list.Find(k0); ok { t.Fail() } if !list.IsEmpty() { t.Fail() } // Test at the beginning of the list. for i := 0; i < maxN; i++ { key := []byte(strconv.Itoa(maxN - i)) list.Insert(key, key) } for i := 0; i < maxN; i++ { key := []byte(strconv.Itoa(maxN - i)) if _, ok, _ := list.Find(key); !ok { t.Fail() } } list = New(memStore) // Test at the end of the list. for i := 0; i < maxN; i++ { key := []byte(strconv.Itoa(i)) list.Insert(key, key) } for i := 0; i < maxN; i++ { key := []byte(strconv.Itoa(i)) if _, ok, _ := list.Find(key); !ok { t.Fail() } } list = New(memStore) // Test at random positions in the list. rList := rand.Perm(maxN) for _, e := range rList { key := []byte(strconv.Itoa(e)) // println("insert", e) list.Insert(key, key) } for _, e := range rList { key := []byte(strconv.Itoa(e)) // println("find", e) if _, ok, _ := list.Find(key); !ok { t.Fail() } } // println("print list") list.println() } func Element(x int) []byte { return []byte(strconv.Itoa(x)) } func TestDelete(t *testing.T) { k0 := []byte("0") var list *SkipList // Delete on empty list list.Delete(k0) list = New(memStore) list.Delete(k0) if !list.IsEmpty() { t.Fail() } list.Insert(k0, k0) list.Delete(k0) if !list.IsEmpty() { t.Fail() } // Delete elements at the beginning of the list. for i := 0; i < maxN; i++ { list.Insert(Element(i), Element(i)) } for i := 0; i < maxN; i++ { list.Delete(Element(i)) } if !list.IsEmpty() { t.Fail() } list = New(memStore) // Delete elements at the end of the list. for i := 0; i < maxN; i++ { list.Insert(Element(i), Element(i)) } for i := 0; i < maxN; i++ { list.Delete(Element(maxN - i - 1)) } if !list.IsEmpty() { t.Fail() } list = New(memStore) // Delete elements at random positions in the list. rList := rand.Perm(maxN) for _, e := range rList { list.Insert(Element(e), Element(e)) } for _, e := range rList { list.Delete(Element(e)) } if !list.IsEmpty() { t.Fail() } } func TestNext(t *testing.T) { list := New(memStore) for i := 0; i < maxN; i++ { list.Insert(Element(i), Element(i)) } smallest, _ := list.GetSmallestNode() largest, _ := list.GetLargestNode() lastNode := smallest node := lastNode for node != largest { node, _ = list.Next(node) // Must always be incrementing here! if bytes.Compare(node.Key, lastNode.Key) <= 0 { t.Fail() } // Next.Prev must always point to itself! prevNode, _ := list.Prev(node) nextNode, _ := list.Next(prevNode) if nextNode != node { t.Fail() } lastNode = node } if nextNode, _ := list.Next(largest); nextNode != smallest { t.Fail() } } func TestPrev(t *testing.T) { list := New(memStore) for i := 0; i < maxN; i++ { list.Insert(Element(i), Element(i)) } smallest, _ := list.GetSmallestNode() largest, _ := list.GetLargestNode() lastNode := largest node := lastNode for node != smallest { node, _ = list.Prev(node) // Must always be incrementing here! if bytes.Compare(node.Key, lastNode.Key) >= 0 { t.Fail() } // Next.Prev must always point to itself! nextNode, _ := list.Next(node) prevNode, _ := list.Prev(nextNode) if prevNode != node { t.Fail() } lastNode = node } if prevNode, _ := list.Prev(smallest); prevNode != largest { t.Fail() } } func TestFindGreaterOrEqual(t *testing.T) { maxNumber := maxN * 100 var list *SkipList var listPointer *SkipList // Test on empty list. if _, ok, _ := listPointer.FindGreaterOrEqual(Element(0)); ok { t.Fail() } list = New(memStore) for i := 0; i < maxN; i++ { list.Insert(Element(rand.Intn(maxNumber)), Element(i)) } for i := 0; i < maxN; i++ { key := Element(rand.Intn(maxNumber)) if v, ok, _ := list.FindGreaterOrEqual(key); ok { // if f is v should be bigger than the element before if v.Prev != nil && bytes.Compare(v.Prev.Key, key) >= 0 { fmt.Printf("PrevV: %s\n key: %s\n\n", string(v.Prev.Key), string(key)) t.Fail() } // v should be bigger or equal to f // If we compare directly, we get an equal key with a difference on the 10th decimal point, which fails. if bytes.Compare(v.Key, key) < 0 { fmt.Printf("v: %s\n key: %s\n\n", string(v.Key), string(key)) t.Fail() } } else { lastNode, _ := list.GetLargestNode() lastV := lastNode.GetValue() // It is OK, to fail, as long as f is bigger than the last element. if bytes.Compare(key, lastV) <= 0 { fmt.Printf("lastV: %s\n key: %s\n\n", string(lastV), string(key)) t.Fail() } } } } func TestChangeValue(t *testing.T) { list := New(memStore) for i := 0; i < maxN; i++ { list.Insert(Element(i), []byte("value")) } for i := 0; i < maxN; i++ { // The key only looks at the int so the string doesn't matter here! f1, ok, _ := list.Find(Element(i)) if !ok { t.Fail() } err := list.ChangeValue(f1, []byte("different value")) if err != nil { t.Fail() } f2, ok, _ := list.Find(Element(i)) if !ok { t.Fail() } if bytes.Compare(f2.GetValue(), []byte("different value")) != 0 { t.Fail() } } }