aboutsummaryrefslogblamecommitdiffstats
path: root/trie/iterator_test.go
blob: 4f633b195fa964582c4f028fa1464c5d7550996c (plain) (tree)
1
2
3
4
5
6
7
8
9
                                         
                                                
  
                                                                                  



                                                                              
                                                                             
                                                                 
                                                               


                                                                           
                                                                                  
 
            
 
        

               
                   




                                                

                                 
                          



                                        



                                                                     
         
                                      
                                  
                                  
                                                         
         
                        
 
                                        
                                                 
                       
                                                        

         
























                                                                                                  
                                                 














                                                            


                 







                                                                               
                                                          

                                                      


                                                         
                                  
                                                        


                                                                                      
                                           






                                                                             




                                                                    
 


                              
                       
                       

                       
                     

                       
                     

 
                      










                          








































                                                                                              

                                           
                                       

                                                          
                         

                           
                                       

                                                          
                         

                                        
                                                                                        
                             











                                                        
                                                                                                                 





                                                                                                 





                                                          
                         




                                                          
                         
 
                                                                                                   
                             


                                       
                               
                               

























                                                                                                                   









                                                                                             



                                                                                                     
                                        


                                           


                                                       



                                              
                                                                          
 








                                        

                                                                
                                               


                                                                                    




                                        
                     





                                                                                 


                                     
                            

                                                    



                                                     




                                                             
                                                    



                                                                           
                            
                                                   


                                                 












                                                                                                    







                                                                     
                                                                          
                                        


                                            


                                                        



                                         
                                                                                                           




                                       

                                                        



                                                           

                                                                                    
                                  






                                                                 
                                     
                    
                                                        


                                                       

















                                                                                   
// Copyright 2014 The go-ethereum Authors
// This file is part of the go-ethereum library.
//
// The go-ethereum library is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// The go-ethereum library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.

package trie

import (
    "bytes"
    "fmt"
    "math/rand"
    "testing"

    "github.com/ethereum/go-ethereum/common"
    "github.com/ethereum/go-ethereum/ethdb"
)

func TestIterator(t *testing.T) {
    trie := newEmpty()
    vals := []struct{ k, v string }{
        {"do", "verb"},
        {"ether", "wookiedoo"},
        {"horse", "stallion"},
        {"shaman", "horse"},
        {"doge", "coin"},
        {"dog", "puppy"},
        {"somethingveryoddindeedthis is", "myothernodedata"},
    }
    all := make(map[string]string)
    for _, val := range vals {
        all[val.k] = val.v
        trie.Update([]byte(val.k), []byte(val.v))
    }
    trie.Commit(nil)

    found := make(map[string]string)
    it := NewIterator(trie.NodeIterator(nil))
    for it.Next() {
        found[string(it.Key)] = string(it.Value)
    }

    for k, v := range all {
        if found[k] != v {
            t.Errorf("iterator value mismatch for %s: got %q want %q", k, found[k], v)
        }
    }
}

type kv struct {
    k, v []byte
    t    bool
}

func TestIteratorLargeData(t *testing.T) {
    trie := newEmpty()
    vals := make(map[string]*kv)

    for i := byte(0); i < 255; i++ {
        value := &kv{common.LeftPadBytes([]byte{i}, 32), []byte{i}, false}
        value2 := &kv{common.LeftPadBytes([]byte{10, i}, 32), []byte{i}, false}
        trie.Update(value.k, value.v)
        trie.Update(value2.k, value2.v)
        vals[string(value.k)] = value
        vals[string(value2.k)] = value2
    }

    it := NewIterator(trie.NodeIterator(nil))
    for it.Next() {
        vals[string(it.Key)].t = true
    }

    var untouched []*kv
    for _, value := range vals {
        if !value.t {
            untouched = append(untouched, value)
        }
    }

    if len(untouched) > 0 {
        t.Errorf("Missed %d nodes", len(untouched))
        for _, value := range untouched {
            t.Error(value)
        }
    }
}

// Tests that the node iterator indeed walks over the entire database contents.
func TestNodeIteratorCoverage(t *testing.T) {
    // Create some arbitrary test trie to iterate
    db, trie, _ := makeTestTrie()

    // Gather all the node hashes found by the iterator
    hashes := make(map[common.Hash]struct{})
    for it := trie.NodeIterator(nil); it.Next(true); {
        if it.Hash() != (common.Hash{}) {
            hashes[it.Hash()] = struct{}{}
        }
    }
    // Cross check the hashes and the database itself
    for hash := range hashes {
        if _, err := db.Node(hash); err != nil {
            t.Errorf("failed to retrieve reported node %x: %v", hash, err)
        }
    }
    for hash, obj := range db.dirties {
        if obj != nil && hash != (common.Hash{}) {
            if _, ok := hashes[hash]; !ok {
                t.Errorf("state entry not reported %x", hash)
            }
        }
    }
    for _, key := range db.diskdb.(*ethdb.MemDatabase).Keys() {
        if _, ok := hashes[common.BytesToHash(key)]; !ok {
            t.Errorf("state entry not reported %x", key)
        }
    }
}

type kvs struct{ k, v string }

var testdata1 = []kvs{
    {"barb", "ba"},
    {"bard", "bc"},
    {"bars", "bb"},
    {"bar", "b"},
    {"fab", "z"},
    {"food", "ab"},
    {"foos", "aa"},
    {"foo", "a"},
}

var testdata2 = []kvs{
    {"aardvark", "c"},
    {"bar", "b"},
    {"barb", "bd"},
    {"bars", "be"},
    {"fab", "z"},
    {"foo", "a"},
    {"foos", "aa"},
    {"food", "ab"},
    {"jars", "d"},
}

func TestIteratorSeek(t *testing.T) {
    trie := newEmpty()
    for _, val := range testdata1 {
        trie.Update([]byte(val.k), []byte(val.v))
    }

    // Seek to the middle.
    it := NewIterator(trie.NodeIterator([]byte("fab")))
    if err := checkIteratorOrder(testdata1[4:], it); err != nil {
        t.Fatal(err)
    }

    // Seek to a non-existent key.
    it = NewIterator(trie.NodeIterator([]byte("barc")))
    if err := checkIteratorOrder(testdata1[1:], it); err != nil {
        t.Fatal(err)
    }

    // Seek beyond the end.
    it = NewIterator(trie.NodeIterator([]byte("z")))
    if err := checkIteratorOrder(nil, it); err != nil {
        t.Fatal(err)
    }
}

func checkIteratorOrder(want []kvs, it *Iterator) error {
    for it.Next() {
        if len(want) == 0 {
            return fmt.Errorf("didn't expect any more values, got key %q", it.Key)
        }
        if !bytes.Equal(it.Key, []byte(want[0].k)) {
            return fmt.Errorf("wrong key: got %q, want %q", it.Key, want[0].k)
        }
        want = want[1:]
    }
    if len(want) > 0 {
        return fmt.Errorf("iterator ended early, want key %q", want[0])
    }
    return nil
}

func TestDifferenceIterator(t *testing.T) {
    triea := newEmpty()
    for _, val := range testdata1 {
        triea.Update([]byte(val.k), []byte(val.v))
    }
    triea.Commit(nil)

    trieb := newEmpty()
    for _, val := range testdata2 {
        trieb.Update([]byte(val.k), []byte(val.v))
    }
    trieb.Commit(nil)

    found := make(map[string]string)
    di, _ := NewDifferenceIterator(triea.NodeIterator(nil), trieb.NodeIterator(nil))
    it := NewIterator(di)
    for it.Next() {
        found[string(it.Key)] = string(it.Value)
    }

    all := []struct{ k, v string }{
        {"aardvark", "c"},
        {"barb", "bd"},
        {"bars", "be"},
        {"jars", "d"},
    }
    for _, item := range all {
        if found[item.k] != item.v {
            t.Errorf("iterator value mismatch for %s: got %v want %v", item.k, found[item.k], item.v)
        }
    }
    if len(found) != len(all) {
        t.Errorf("iterator count mismatch: got %d values, want %d", len(found), len(all))
    }
}

func TestUnionIterator(t *testing.T) {
    triea := newEmpty()
    for _, val := range testdata1 {
        triea.Update([]byte(val.k), []byte(val.v))
    }
    triea.Commit(nil)

    trieb := newEmpty()
    for _, val := range testdata2 {
        trieb.Update([]byte(val.k), []byte(val.v))
    }
    trieb.Commit(nil)

    di, _ := NewUnionIterator([]NodeIterator{triea.NodeIterator(nil), trieb.NodeIterator(nil)})
    it := NewIterator(di)

    all := []struct{ k, v string }{
        {"aardvark", "c"},
        {"barb", "ba"},
        {"barb", "bd"},
        {"bard", "bc"},
        {"bars", "bb"},
        {"bars", "be"},
        {"bar", "b"},
        {"fab", "z"},
        {"food", "ab"},
        {"foos", "aa"},
        {"foo", "a"},
        {"jars", "d"},
    }

    for i, kv := range all {
        if !it.Next() {
            t.Errorf("Iterator ends prematurely at element %d", i)
        }
        if kv.k != string(it.Key) {
            t.Errorf("iterator value mismatch for element %d: got key %s want %s", i, it.Key, kv.k)
        }
        if kv.v != string(it.Value) {
            t.Errorf("iterator value mismatch for element %d: got value %s want %s", i, it.Value, kv.v)
        }
    }
    if it.Next() {
        t.Errorf("Iterator returned extra values.")
    }
}

func TestIteratorNoDups(t *testing.T) {
    var tr Trie
    for _, val := range testdata1 {
        tr.Update([]byte(val.k), []byte(val.v))
    }
    checkIteratorNoDups(t, tr.NodeIterator(nil), nil)
}

// This test checks that nodeIterator.Next can be retried after inserting missing trie nodes.
func TestIteratorContinueAfterErrorDisk(t *testing.T)    { testIteratorContinueAfterError(t, false) }
func TestIteratorContinueAfterErrorMemonly(t *testing.T) { testIteratorContinueAfterError(t, true) }

func testIteratorContinueAfterError(t *testing.T, memonly bool) {
    diskdb := ethdb.NewMemDatabase()
    triedb := NewDatabase(diskdb)

    tr, _ := New(common.Hash{}, triedb)
    for _, val := range testdata1 {
        tr.Update([]byte(val.k), []byte(val.v))
    }
    tr.Commit(nil)
    if !memonly {
        triedb.Commit(tr.Hash(), true)
    }
    wantNodeCount := checkIteratorNoDups(t, tr.NodeIterator(nil), nil)

    var (
        diskKeys [][]byte
        memKeys  []common.Hash
    )
    if memonly {
        memKeys = triedb.Nodes()
    } else {
        diskKeys = diskdb.Keys()
    }
    for i := 0; i < 20; i++ {
        // Create trie that will load all nodes from DB.
        tr, _ := New(tr.Hash(), triedb)

        // Remove a random node from the database. It can't be the root node
        // because that one is already loaded.
        var (
            rkey common.Hash
            rval []byte
            robj *cachedNode
        )
        for {
            if memonly {
                rkey = memKeys[rand.Intn(len(memKeys))]
            } else {
                copy(rkey[:], diskKeys[rand.Intn(len(diskKeys))])
            }
            if rkey != tr.Hash() {
                break
            }
        }
        if memonly {
            robj = triedb.dirties[rkey]
            delete(triedb.dirties, rkey)
        } else {
            rval, _ = diskdb.Get(rkey[:])
            diskdb.Delete(rkey[:])
        }
        // Iterate until the error is hit.
        seen := make(map[string]bool)
        it := tr.NodeIterator(nil)
        checkIteratorNoDups(t, it, seen)
        missing, ok := it.Error().(*MissingNodeError)
        if !ok || missing.NodeHash != rkey {
            t.Fatal("didn't hit missing node, got", it.Error())
        }

        // Add the node back and continue iteration.
        if memonly {
            triedb.dirties[rkey] = robj
        } else {
            diskdb.Put(rkey[:], rval)
        }
        checkIteratorNoDups(t, it, seen)
        if it.Error() != nil {
            t.Fatal("unexpected error", it.Error())
        }
        if len(seen) != wantNodeCount {
            t.Fatal("wrong node iteration count, got", len(seen), "want", wantNodeCount)
        }
    }
}

// Similar to the test above, this one checks that failure to create nodeIterator at a
// certain key prefix behaves correctly when Next is called. The expectation is that Next
// should retry seeking before returning true for the first time.
func TestIteratorContinueAfterSeekErrorDisk(t *testing.T) {
    testIteratorContinueAfterSeekError(t, false)
}
func TestIteratorContinueAfterSeekErrorMemonly(t *testing.T) {
    testIteratorContinueAfterSeekError(t, true)
}

func testIteratorContinueAfterSeekError(t *testing.T, memonly bool) {
    // Commit test trie to db, then remove the node containing "bars".
    diskdb := ethdb.NewMemDatabase()
    triedb := NewDatabase(diskdb)

    ctr, _ := New(common.Hash{}, triedb)
    for _, val := range testdata1 {
        ctr.Update([]byte(val.k), []byte(val.v))
    }
    root, _ := ctr.Commit(nil)
    if !memonly {
        triedb.Commit(root, true)
    }
    barNodeHash := common.HexToHash("05041990364eb72fcb1127652ce40d8bab765f2bfe53225b1170d276cc101c2e")
    var (
        barNodeBlob []byte
        barNodeObj  *cachedNode
    )
    if memonly {
        barNodeObj = triedb.dirties[barNodeHash]
        delete(triedb.dirties, barNodeHash)
    } else {
        barNodeBlob, _ = diskdb.Get(barNodeHash[:])
        diskdb.Delete(barNodeHash[:])
    }
    // Create a new iterator that seeks to "bars". Seeking can't proceed because
    // the node is missing.
    tr, _ := New(root, triedb)
    it := tr.NodeIterator([]byte("bars"))
    missing, ok := it.Error().(*MissingNodeError)
    if !ok {
        t.Fatal("want MissingNodeError, got", it.Error())
    } else if missing.NodeHash != barNodeHash {
        t.Fatal("wrong node missing")
    }
    // Reinsert the missing node.
    if memonly {
        triedb.dirties[barNodeHash] = barNodeObj
    } else {
        diskdb.Put(barNodeHash[:], barNodeBlob)
    }
    // Check that iteration produces the right set of values.
    if err := checkIteratorOrder(testdata1[2:], NewIterator(it)); err != nil {
        t.Fatal(err)
    }
}

func checkIteratorNoDups(t *testing.T, it NodeIterator, seen map[string]bool) int {
    if seen == nil {
        seen = make(map[string]bool)
    }
    for it.Next(true) {
        if seen[string(it.Path())] {
            t.Fatalf("iterator visited node path %x twice", it.Path())
        }
        seen[string(it.Path())] = true
    }
    return len(seen)
}