iterator.go 3.22 KB
Newer Older
1
// Copyright 2014 The go-ethereum Authors
2
// This file is part of the go-ethereum library.
3
//
4
// The go-ethereum library is free software: you can redistribute it and/or modify
5 6 7 8
// 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.
//
9
// The go-ethereum library is distributed in the hope that it will be useful,
10
// but WITHOUT ANY WARRANTY; without even the implied warranty of
11
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 13 14
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
15
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
16

obscuren's avatar
obscuren committed
17
package trie
obscuren's avatar
obscuren committed
18

19 20 21 22 23 24
import (
	"bytes"

	"github.com/ethereum/go-ethereum/logger"
	"github.com/ethereum/go-ethereum/logger/glog"
)
obscuren's avatar
obscuren committed
25 26 27 28 29

type Iterator struct {
	trie *Trie

	Key   []byte
30
	Value []byte
obscuren's avatar
obscuren committed
31 32 33
}

func NewIterator(trie *Trie) *Iterator {
34
	return &Iterator{trie: trie, Key: nil}
obscuren's avatar
obscuren committed
35 36
}

37
func (self *Iterator) Next() bool {
38 39 40 41 42 43
	isIterStart := false
	if self.Key == nil {
		isIterStart = true
		self.Key = make([]byte, 32)
	}

Felix Lange's avatar
Felix Lange committed
44
	key := remTerm(compactHexDecode(self.Key))
45
	k := self.next(self.trie.root, key, isIterStart)
obscuren's avatar
obscuren committed
46

Felix Lange's avatar
Felix Lange committed
47
	self.Key = []byte(decodeCompact(k))
obscuren's avatar
obscuren committed
48

49
	return len(k) > 0
obscuren's avatar
obscuren committed
50 51
}

Felix Lange's avatar
Felix Lange committed
52
func (self *Iterator) next(node interface{}, key []byte, isIterStart bool) []byte {
53
	if node == nil {
obscuren's avatar
obscuren committed
54
		return nil
55
	}
obscuren's avatar
obscuren committed
56

57
	switch node := node.(type) {
Felix Lange's avatar
Felix Lange committed
58
	case fullNode:
59
		if len(key) > 0 {
Felix Lange's avatar
Felix Lange committed
60
			k := self.next(node[key[0]], key[1:], isIterStart)
61 62
			if k != nil {
				return append([]byte{key[0]}, k...)
obscuren's avatar
obscuren committed
63 64 65
			}
		}

66
		var r byte
obscuren's avatar
obscuren committed
67 68 69 70 71
		if len(key) > 0 {
			r = key[0] + 1
		}

		for i := r; i < 16; i++ {
Felix Lange's avatar
Felix Lange committed
72
			k := self.key(node[i])
73 74
			if k != nil {
				return append([]byte{i}, k...)
obscuren's avatar
obscuren committed
75 76 77
			}
		}

Felix Lange's avatar
Felix Lange committed
78 79 80
	case shortNode:
		k := remTerm(node.Key)
		if vnode, ok := node.Val.(valueNode); ok {
81 82 83
			switch bytes.Compare([]byte(k), key) {
			case 0:
				if isIterStart {
Felix Lange's avatar
Felix Lange committed
84
					self.Value = vnode
85 86 87
					return k
				}
			case 1:
Felix Lange's avatar
Felix Lange committed
88
				self.Value = vnode
obscuren's avatar
obscuren committed
89 90 91
				return k
			}
		} else {
Felix Lange's avatar
Felix Lange committed
92
			cnode := node.Val
93

obscuren's avatar
obscuren committed
94
			var ret []byte
95
			skey := key[len(k):]
Felix Lange's avatar
Felix Lange committed
96
			if bytes.HasPrefix(key, k) {
97
				ret = self.next(cnode, skey, isIterStart)
obscuren's avatar
obscuren committed
98
			} else if bytes.Compare(k, key[:len(k)]) > 0 {
obscuren's avatar
obscuren committed
99
				return self.key(node)
obscuren's avatar
obscuren committed
100 101 102 103 104 105 106
			}

			if ret != nil {
				return append(k, ret...)
			}
		}

Felix Lange's avatar
Felix Lange committed
107
	case hashNode:
108 109 110 111 112
		rn, err := self.trie.resolveHash(node, nil, nil)
		if err != nil && glog.V(logger.Error) {
			glog.Errorf("Unhandled trie error: %v", err)
		}
		return self.next(rn, key, isIterStart)
Felix Lange's avatar
Felix Lange committed
113
	}
obscuren's avatar
obscuren committed
114 115 116
	return nil
}

Felix Lange's avatar
Felix Lange committed
117
func (self *Iterator) key(node interface{}) []byte {
118
	switch node := node.(type) {
Felix Lange's avatar
Felix Lange committed
119
	case shortNode:
120
		// Leaf node
Felix Lange's avatar
Felix Lange committed
121 122 123
		k := remTerm(node.Key)
		if vnode, ok := node.Val.(valueNode); ok {
			self.Value = vnode
124 125
			return k
		}
Felix Lange's avatar
Felix Lange committed
126 127 128 129
		return append(k, self.key(node.Val)...)
	case fullNode:
		if node[16] != nil {
			self.Value = node[16].(valueNode)
130 131 132
			return []byte{16}
		}
		for i := 0; i < 16; i++ {
Felix Lange's avatar
Felix Lange committed
133
			k := self.key(node[i])
134 135 136 137
			if k != nil {
				return append([]byte{byte(i)}, k...)
			}
		}
Felix Lange's avatar
Felix Lange committed
138
	case hashNode:
139 140 141 142 143
		rn, err := self.trie.resolveHash(node, nil, nil)
		if err != nil && glog.V(logger.Error) {
			glog.Errorf("Unhandled trie error: %v", err)
		}
		return self.key(rn)
144 145 146
	}

	return nil
obscuren's avatar
obscuren committed
147
}