-
Notifications
You must be signed in to change notification settings - Fork 70
/
Copy pathnibbles.go
108 lines (91 loc) · 2.16 KB
/
nibbles.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
package main
import (
"fmt"
)
type Nibble byte
func IsNibble(nibble byte) bool {
n := int(nibble)
// 0-9 && a-f
return n >= 0 && n < 16
}
func FromNibbleByte(n byte) (Nibble, error) {
if !IsNibble(n) {
return 0, fmt.Errorf("non-nibble byte: %v", n)
}
return Nibble(n), nil
}
// nibbles contain one nibble per byte
func FromNibbleBytes(nibbles []byte) ([]Nibble, error) {
ns := make([]Nibble, 0, len(nibbles))
for _, n := range nibbles {
nibble, err := FromNibbleByte(n)
if err != nil {
return nil, fmt.Errorf("contains non-nibble byte: %w", err)
}
ns = append(ns, nibble)
}
return ns, nil
}
func FromByte(b byte) []Nibble {
return []Nibble{
Nibble(byte(b >> 4)),
Nibble(byte(b % 16)),
}
}
func FromBytes(bs []byte) []Nibble {
ns := make([]Nibble, 0, len(bs)*2)
for _, b := range bs {
ns = append(ns, FromByte(b)...)
}
return ns
}
func FromString(s string) []Nibble {
return FromBytes([]byte(s))
}
// ToPrefixed add nibble prefix to a slice of nibbles to make its length even
// the prefix indicts whether a node is a leaf node.
func ToPrefixed(ns []Nibble, isLeafNode bool) []Nibble {
// create prefix
var prefixBytes []Nibble
// odd number of nibbles
if len(ns)%2 > 0 {
prefixBytes = []Nibble{1}
} else {
// even number of nibbles
prefixBytes = []Nibble{0, 0}
}
// append prefix to all nibble bytes
prefixed := make([]Nibble, 0, len(prefixBytes)+len(ns))
prefixed = append(prefixed, prefixBytes...)
prefixed = append(prefixed, ns...)
// update prefix if is leaf node
if isLeafNode {
prefixed[0] += 2
}
return prefixed
}
// ToBytes converts a slice of nibbles to a byte slice
// assuming the nibble slice has even number of nibbles.
func ToBytes(ns []Nibble) []byte {
buf := make([]byte, 0, len(ns)/2)
for i := 0; i < len(ns); i += 2 {
b := byte(ns[i]<<4) + byte(ns[i+1])
buf = append(buf, b)
}
return buf
}
// [0,1,2,3], [0,1,2] => 3
// [0,1,2,3], [0,1,2,3] => 4
// [0,1,2,3], [0,1,2,3,4] => 4
func PrefixMatchedLen(node1 []Nibble, node2 []Nibble) int {
matched := 0
for i := 0; i < len(node1) && i < len(node2); i++ {
n1, n2 := node1[i], node2[i]
if n1 == n2 {
matched++
} else {
break
}
}
return matched
}