diff options
author | Maran <maran.hidskes@gmail.com> | 2014-05-27 16:29:47 +0800 |
---|---|---|
committer | Maran <maran.hidskes@gmail.com> | 2014-05-27 16:29:47 +0800 |
commit | 817def000ba6488b3999fa9b8eac9fce43984f19 (patch) | |
tree | 4f5f0efa5ae670bc658ef0bce123a44a0644f1f5 /ethutil/trie.go | |
parent | 2232974cda9ed73c62a370545742ef88332c5f1e (diff) | |
parent | 6e24d603a157ba6f66d49132d16221d9adbdff4d (diff) | |
download | dexon-817def000ba6488b3999fa9b8eac9fce43984f19.tar dexon-817def000ba6488b3999fa9b8eac9fce43984f19.tar.gz dexon-817def000ba6488b3999fa9b8eac9fce43984f19.tar.bz2 dexon-817def000ba6488b3999fa9b8eac9fce43984f19.tar.lz dexon-817def000ba6488b3999fa9b8eac9fce43984f19.tar.xz dexon-817def000ba6488b3999fa9b8eac9fce43984f19.tar.zst dexon-817def000ba6488b3999fa9b8eac9fce43984f19.zip |
Merge branch 'develop' of github.com:ethereum/eth-go into develop
Diffstat (limited to 'ethutil/trie.go')
-rw-r--r-- | ethutil/trie.go | 46 |
1 files changed, 46 insertions, 0 deletions
diff --git a/ethutil/trie.go b/ethutil/trie.go index c993e4d8f..18d0a5f0a 100644 --- a/ethutil/trie.go +++ b/ethutil/trie.go @@ -442,6 +442,8 @@ type TrieIterator struct { shas [][]byte values []string + + lastNode []byte } func (t *Trie) NewIterator() *TrieIterator { @@ -513,3 +515,47 @@ func (it *TrieIterator) Key() string { func (it *TrieIterator) Value() string { return "" } + +type EachCallback func(key string, node *Value) + +func (it *TrieIterator) Each(cb EachCallback) { + it.fetchNode(nil, NewValue(it.trie.Root).Bytes(), cb) +} + +func (it *TrieIterator) fetchNode(key []int, node []byte, cb EachCallback) { + it.iterateNode(key, it.trie.cache.Get(node), cb) +} + +func (it *TrieIterator) iterateNode(key []int, currentNode *Value, cb EachCallback) { + if currentNode.Len() == 2 { + k := CompactDecode(currentNode.Get(0).Str()) + + if currentNode.Get(1).Str() == "" { + it.iterateNode(key, currentNode.Get(1), cb) + } else { + pk := append(key, k...) + + if k[len(k)-1] == 16 { + cb(DecodeCompact(pk), currentNode.Get(1)) + } else { + it.fetchNode(pk, currentNode.Get(1).Bytes(), cb) + } + } + } else { + for i := 0; i < currentNode.Len(); i++ { + pk := append(key, i) + if i == 16 && currentNode.Get(i).Len() != 0 { + cb(DecodeCompact(pk), currentNode.Get(i)) + } else { + if currentNode.Get(i).Str() == "" { + it.iterateNode(pk, currentNode.Get(i), cb) + } else { + val := currentNode.Get(i).Str() + if val != "" { + it.fetchNode(pk, []byte(val), cb) + } + } + } + } + } +} |