-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathtrie.go
157 lines (126 loc) · 3.21 KB
/
trie.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
package mpt
import (
"bytes"
"github.com/begmaroman/mpt/enc"
"github.com/begmaroman/mpt/node"
)
var (
nilNode = enc.HexToHash("56e81f171bcc55a6ff8345e692c0f86e5b48e01b996cadc001622fb5e363b421")
)
type Trie struct {
db *Database
node node.Node
}
func NewTrie(node node.Node) *Trie {
return &Trie{
node: node,
}
}
func NewEmptyTrie() *Trie {
return &Trie{}
}
// Hash calculate and return hash of the tree
func (t *Trie) Hash() (enc.Hash, error) {
bHash, cached, err := t.hashRoot(nil)
hash := enc.BytesToHash(bHash.(node.HashNode))
if err == nil {
t.node = cached
}
return hash, err
}
// Put inserts key/value pair into tree
func (t *Trie) Put(key, value []byte) bool {
nd, ok := t.put(t.node, enc.BytesToHex(key), node.NewLeafNode(value), nil)
if !ok {
return false
}
t.node = nd
return true
}
// Get returns value for incoming key
func (t *Trie) Get(key []byte) ([]byte, bool) {
val, nd, resolved := t.get(t.node, enc.BytesToHex(key))
if resolved {
t.node = nd
}
return val, resolved
}
// Delete remove key/value par from chain
func (t *Trie) Delete(key []byte) bool {
n, ok := t.delete(t.node, enc.BytesToHex(key), nil)
if !ok {
return false
}
t.node = n
return true
}
// Update update value by key
func (t *Trie) Update(key, value []byte) bool {
var n node.Node
var ok bool
kHex := enc.BytesToHex(key)
n, ok = t.delete(t.node, kHex, nil)
if !ok {
return false
}
if len(value) != 0 {
n, ok = t.put(n, kHex, node.NewLeafNode(value), nil)
if !ok {
return false
}
}
t.node = n
return true
}
// put add new data to the node
func (t *Trie) put(n node.Node, key []byte, value node.Node, prefix []byte) (node.Node, bool) {
if len(key) == 0 {
if val, ok := n.(node.LeafNode); ok {
return value, !bytes.Equal(val, value.(node.LeafNode))
}
return value, true
}
if n == nil {
// initialize new ExtensionNode and set key/value pair
return node.NewExtensionNode(key, value), true
}
return n.Put(key, value)
}
// get returns data from node by key
func (t *Trie) get(n node.Node, key []byte) ([]byte, node.Node, bool) {
if n == nil {
return nil, nil, false
}
return n.Find(key)
}
// delete remove daa from node by key
func (t *Trie) delete(n node.Node, key, prefix []byte) (node.Node, bool) {
if n == nil {
return nil, false
}
return n.Delete(key)
}
// hashToot do hash for root of tree
func (t *Trie) hashRoot(db *Database) (node.Node, node.Node, error) {
if t.node == nil {
return node.NewHashNode(nilNode.Bytes()), nil, nil
}
h := NewEncryptor()
defer hPool.Put(h)
return h.hash(t.node, db, true)
}
// resolve returns results of resolveHash if incoming node.Node is HashNode, or return incoming node.Node
func (t *Trie) resolve(n node.Node, prefix []byte) (node.Node, error) {
if n, ok := n.(node.HashNode); ok {
return t.resolveHash(n, prefix)
}
return n, nil
}
// resolveHash return node.Node from DB or error if not found
func (t *Trie) resolveHash(n node.HashNode, prefix []byte) (node.Node, error) {
hash := n.Hash() // get hash of HashNode
if nd := t.db.GetNode(hash); nd != nil { // lookup node.Node in database
return nd, nil // return node.Node from database if exist
}
return nil, NewErrNodeNotFound(prefix, hash) // return error if not found
}