-
Notifications
You must be signed in to change notification settings - Fork 330
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[db]log mptrie node #3634
[db]log mptrie node #3634
Changes from 24 commits
fd984d8
dc8b725
ee087b0
fc7ddd6
3a91c4c
7cd23c4
0acee52
549a8ec
76e00cf
b54aaca
af14413
6f736e4
12fd4cf
c7bbe1e
c9b5792
23622ed
8166a5d
6521e42
a575201
33ed183
99efe1a
d485731
c501416
715ccb7
070a33c
1b1944d
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -46,6 +46,9 @@ func newBranchNode( | |
} | ||
} | ||
} | ||
if err := logNode(_nodeTypeBranch, _actionTypeNew, bnode, cli); err != nil { | ||
return nil, err | ||
} | ||
return bnode, nil | ||
} | ||
|
||
|
@@ -69,6 +72,9 @@ func newRootBranchNode(cli client, children map[byte]node, indices *SortedList, | |
} | ||
} | ||
} | ||
if err := logNode(_nodeTypeBranch, _actionTypeNew, bnode, cli); err != nil { | ||
return nil, err | ||
} | ||
return bnode, nil | ||
} | ||
|
||
|
@@ -85,6 +91,9 @@ func newBranchNodeFromProtoPb(pb *triepb.BranchPb, hashVal []byte) *branchNode { | |
} | ||
bnode.indices = NewSortedList(bnode.children) | ||
bnode.cacheNode.serializable = bnode | ||
if err := logNode(_nodeTypeBranch, _actionTypeNew, bnode, nil); err != nil { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. not working here |
||
panic(err) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. why is There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
|
||
} | ||
return bnode | ||
} | ||
|
||
|
@@ -101,6 +110,9 @@ func (b *branchNode) Children() []node { | |
} | ||
|
||
func (b *branchNode) Delete(cli client, key keyType, offset uint8) (node, error) { | ||
if err := logNode(_nodeTypeBranch, _actionTypeDelete, b, cli); err != nil { | ||
return nil, err | ||
} | ||
offsetKey := key[offset] | ||
child, err := b.child(offsetKey) | ||
if err != nil { | ||
|
@@ -154,6 +166,9 @@ func (b *branchNode) Delete(cli client, key keyType, offset uint8) (node, error) | |
} | ||
|
||
func (b *branchNode) Upsert(cli client, key keyType, offset uint8, value []byte) (node, error) { | ||
if err := logNode(_nodeTypeBranch, _actionTypeUpsert, b, cli); err != nil { | ||
return nil, err | ||
} | ||
var newChild node | ||
offsetKey := key[offset] | ||
child, err := b.child(offsetKey) | ||
|
@@ -171,6 +186,9 @@ func (b *branchNode) Upsert(cli client, key keyType, offset uint8, value []byte) | |
} | ||
|
||
func (b *branchNode) Search(cli client, key keyType, offset uint8) (node, error) { | ||
if err := logNode(_nodeTypeBranch, _actionTypeSearch, b, cli); err != nil { | ||
return nil, err | ||
} | ||
child, err := b.child(key[offset]) | ||
if err != nil { | ||
return nil, err | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -38,6 +38,9 @@ func newExtensionNode( | |
return nil, err | ||
} | ||
} | ||
if err := logNode(_nodeTypeExtension, _actionTypeNew, e, cli); err != nil { | ||
return nil, err | ||
} | ||
return e, nil | ||
} | ||
|
||
|
@@ -51,10 +54,16 @@ func newExtensionNodeFromProtoPb(pb *triepb.ExtendPb, hashVal []byte) *extension | |
child: newHashNode(pb.Value), | ||
} | ||
e.cacheNode.serializable = e | ||
if err := logNode(_nodeTypeExtension, _actionTypeNew, e, nil); err != nil { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. not working here |
||
panic(err) | ||
} | ||
return e | ||
} | ||
|
||
func (e *extensionNode) Delete(cli client, key keyType, offset uint8) (node, error) { | ||
if err := logNode(_nodeTypeExtension, _actionTypeDelete, e, cli); err != nil { | ||
return nil, err | ||
} | ||
matched := e.commonPrefixLength(key[offset:]) | ||
if matched != uint8(len(e.path)) { | ||
return nil, trie.ErrNotExist | ||
|
@@ -85,6 +94,9 @@ func (e *extensionNode) Delete(cli client, key keyType, offset uint8) (node, err | |
} | ||
|
||
func (e *extensionNode) Upsert(cli client, key keyType, offset uint8, value []byte) (node, error) { | ||
if err := logNode(_nodeTypeExtension, _actionTypeUpsert, e, cli); err != nil { | ||
return nil, err | ||
} | ||
matched := e.commonPrefixLength(key[offset:]) | ||
if matched == uint8(len(e.path)) { | ||
newChild, err := e.child.Upsert(cli, key, offset+matched, value) | ||
|
@@ -120,6 +132,9 @@ func (e *extensionNode) Upsert(cli client, key keyType, offset uint8, value []by | |
} | ||
|
||
func (e *extensionNode) Search(cli client, key keyType, offset uint8) (node, error) { | ||
if err := logNode(_nodeTypeExtension, _actionTypeSearch, e, cli); err != nil { | ||
return nil, err | ||
} | ||
matched := e.commonPrefixLength(key[offset:]) | ||
if matched != uint8(len(e.path)) { | ||
return nil, trie.ErrNotExist | ||
|
Original file line number | Diff line number | Diff line change | ||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -38,6 +38,9 @@ func newLeafNode( | |||||||||||||||||
return nil, err | ||||||||||||||||||
} | ||||||||||||||||||
} | ||||||||||||||||||
if err := logNode(_nodeTypeLeaf, _actionTypeNew, l, cli); err != nil { | ||||||||||||||||||
return nil, err | ||||||||||||||||||
} | ||||||||||||||||||
return l, nil | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
|
@@ -51,6 +54,9 @@ func newLeafNodeFromProtoPb(pb *triepb.LeafPb, hashVal []byte) *leafNode { | |||||||||||||||||
value: pb.Value, | ||||||||||||||||||
} | ||||||||||||||||||
l.cacheNode.serializable = l | ||||||||||||||||||
if err := logNode(_nodeTypeLeaf, _actionTypeNew, l, nil); err != nil { | ||||||||||||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. not working here There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I thought so too at first. iotex-core/db/trie/mptrie/cachenode.go Lines 25 to 32 in 715ccb7
|
||||||||||||||||||
panic(err) | ||||||||||||||||||
} | ||||||||||||||||||
return l | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
|
@@ -63,13 +69,19 @@ func (l *leafNode) Value() []byte { | |||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (l *leafNode) Delete(cli client, key keyType, offset uint8) (node, error) { | ||||||||||||||||||
if err := logNode(_nodeTypeLeaf, _actionTypeDelete, l, cli); err != nil { | ||||||||||||||||||
return nil, err | ||||||||||||||||||
} | ||||||||||||||||||
if !bytes.Equal(l.key[offset:], key[offset:]) { | ||||||||||||||||||
return nil, trie.ErrNotExist | ||||||||||||||||||
} | ||||||||||||||||||
return nil, l.delete(cli) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (l *leafNode) Upsert(cli client, key keyType, offset uint8, value []byte) (node, error) { | ||||||||||||||||||
if err := logNode(_nodeTypeLeaf, _actionTypeUpsert, l, cli); err != nil { | ||||||||||||||||||
return nil, err | ||||||||||||||||||
} | ||||||||||||||||||
matched := commonPrefixLength(l.key[offset:], key[offset:]) | ||||||||||||||||||
if offset+matched == uint8(len(key)) { | ||||||||||||||||||
if err := l.delete(cli); err != nil { | ||||||||||||||||||
|
@@ -106,7 +118,10 @@ func (l *leafNode) Upsert(cli client, key keyType, offset uint8, value []byte) ( | |||||||||||||||||
return newExtensionNode(cli, l.key[offset:offset+matched], bnode) | ||||||||||||||||||
} | ||||||||||||||||||
|
||||||||||||||||||
func (l *leafNode) Search(_ client, key keyType, offset uint8) (node, error) { | ||||||||||||||||||
func (l *leafNode) Search(cli client, key keyType, offset uint8) (node, error) { | ||||||||||||||||||
if err := logNode(_nodeTypeLeaf, _actionTypeSearch, l, cli); err != nil { | ||||||||||||||||||
return nil, err | ||||||||||||||||||
} | ||||||||||||||||||
if !bytes.Equal(l.key[offset:], key[offset:]) { | ||||||||||||||||||
return nil, trie.ErrNotExist | ||||||||||||||||||
} | ||||||||||||||||||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,127 @@ | ||
package mptrie | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. IMO, the name |
||
|
||
import ( | ||
"bufio" | ||
"os" | ||
|
||
"github.com/pkg/errors" | ||
) | ||
|
||
var ( | ||
enabledLogMptrie = false | ||
logFile *os.File | ||
logWriter *bufio.Writer | ||
) | ||
|
||
type nodeType byte | ||
type actionType byte | ||
|
||
const ( | ||
_nodeTypeLeaf nodeType = 'l' | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Could a byte array |
||
_nodeTypeExtension nodeType = 'e' | ||
_nodeTypeBranch nodeType = 'b' | ||
|
||
_actionTypeSearch actionType = 's' | ||
_actionTypeUpsert actionType = 'u' | ||
_actionTypeDelete actionType = 'd' | ||
_actionTypeNew actionType = 'n' | ||
) | ||
|
||
// nodeEvent is the event of node | ||
type nodeEvent struct { | ||
NodeType nodeType | ||
ActionType actionType | ||
KeyLen uint8 | ||
Key []byte | ||
PathLen uint8 | ||
Path []byte | ||
ChildrenLen uint8 | ||
Children []byte | ||
HashLen uint8 | ||
HashVal []byte | ||
} | ||
|
||
// Bytes returns the bytes of node event | ||
func (e nodeEvent) Bytes() []byte { | ||
b := make([]byte, 0, 1+1+1+e.KeyLen+1+e.PathLen+1+e.ChildrenLen+1+e.HashLen) | ||
b = append(b, byte(e.NodeType)) | ||
b = append(b, byte(e.ActionType)) | ||
b = append(b, e.KeyLen) | ||
b = append(b, e.Key...) | ||
b = append(b, e.PathLen) | ||
b = append(b, e.Path...) | ||
b = append(b, e.ChildrenLen) | ||
b = append(b, e.Children...) | ||
b = append(b, e.HashLen) | ||
b = append(b, e.HashVal...) | ||
return b | ||
} | ||
|
||
// OpenLogDB open the log DB file | ||
func OpenLogDB(dbPath string) error { | ||
var err error | ||
logFile, err = os.OpenFile(dbPath, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0666) | ||
if err != nil { | ||
return err | ||
} | ||
logWriter = bufio.NewWriter(logFile) | ||
enabledLogMptrie = true | ||
return nil | ||
} | ||
|
||
// CloseLogDB close the log DB file | ||
func CloseLogDB() error { | ||
if !enabledLogMptrie { | ||
return nil | ||
} | ||
if err := logWriter.Flush(); err != nil { | ||
return err | ||
} | ||
return logFile.Close() | ||
} | ||
|
||
func logNode(nt nodeType, at actionType, n node, cli client) error { | ||
if !enabledLogMptrie { | ||
return nil | ||
} | ||
nodeKey, nodePath, nodeChildren, hashvalue, err := parseNode(n, cli) | ||
if err != nil { | ||
return err | ||
} | ||
event := nodeEvent{ | ||
NodeType: nt, | ||
ActionType: at, | ||
KeyLen: uint8(len(nodeKey)), | ||
Key: nodeKey, | ||
PathLen: uint8(len(nodePath)), | ||
Path: nodePath, | ||
ChildrenLen: uint8(len(nodeChildren)), | ||
Children: nodeChildren, | ||
HashLen: uint8(len(hashvalue)), | ||
HashVal: hashvalue, | ||
} | ||
// write event length | ||
if err = logWriter.WriteByte(byte(len(event.Bytes()))); err != nil { | ||
return err | ||
} | ||
// write event body | ||
_, err = logWriter.Write(event.Bytes()) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. How to read the binary data in the db? Do we have the tool or component to print the data? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. A byte is used to save the size of the binrary. Binary itself is not readable and needs to be processed by tool. In the lognode_test.go, there is a parseNodeEvent that can parse the binrary. How the tools are provided needs to be discussed (print, database) |
||
return err | ||
} | ||
|
||
func parseNode(n node, cli client) (nodeKey, nodePath, nodeChildren, hashvalue []byte, err error) { | ||
switch n := n.(type) { | ||
case *leafNode: | ||
nodeKey = n.key | ||
hashvalue, err = n.cacheNode.Hash(cli) | ||
case *extensionNode: | ||
nodePath = n.path | ||
hashvalue, err = n.cacheNode.Hash(cli) | ||
case *branchNode: | ||
nodeChildren = n.indices.List() | ||
hashvalue, err = n.cacheNode.Hash(cli) | ||
default: | ||
err = errors.Errorf("unknown node type %T", n) | ||
} | ||
return | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
why is
logging
used at the end of function instead of at the head?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Wait for the
bnode
Initialized