-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcow_trie.go
124 lines (98 loc) · 2.32 KB
/
cow_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
package main
// implementation of a copy-on-write trie data structure
type COWNode struct {
Value string
Next map[rune]*COWNode
}
func NewCOWNode() *COWNode {
return &COWNode{
Value: "",
Next: make(map[rune]*COWNode),
}
}
// Clone node value and all the children next pointers map
func (node *COWNode) Clone() *COWNode {
clone := &COWNode{
Value: node.Value,
Next: make(map[rune]*COWNode),
}
for k, v := range node.Next {
clone.Next[k] = v
}
return clone
}
// Sets key - value pair recursively creating a new cloned node along the way
func (node *COWNode) Set(key string, value string) (*COWNode, error) {
if key == "" {
node.Value = value
return node, nil
}
current := node
runeStr := []rune(key)
r := runeStr[0]
currentClone := current.Clone()
next, ok := currentClone.Next[r]
if !ok {
next = NewCOWNode()
}
next, err := next.Set(string(runeStr[1:]), value)
if err != nil {
return nil, err
}
currentClone.Next[r] = next
return currentClone, nil
}
// Get method
func (node *COWNode) Get(key string) (string, error) {
current := node
for _, r := range key {
next, ok := current.Next[r]
if !ok {
return "", ErrKeyNotFound
}
current = next
}
if current.Value != "" {
return current.Value, nil
}
return "", ErrKeyNotFound
}
func (node *COWNode) Delete(key string) (*COWNode, error) {
return deleteCOWNode(node, key, nil, 0)
}
// Delete function. Works recursively and deletes nodes if they become empty
// An empty node contains no children map pointers and empty value
func deleteCOWNode(node *COWNode, key string, parent *COWNode, parentKey rune) (*COWNode, error) {
if key == "" {
if parent == nil {
return nil, ErrParentDelNotAllowed
}
nodeClone := node.Clone()
nodeClone.Value = ""
if len(nodeClone.Next) == 0 {
delete(parent.Next, parentKey)
return nil, nil
}
return nodeClone, nil
}
runeStr := []rune(key)
r := runeStr[0]
nodeClone := node.Clone()
next, ok := node.Next[r]
if !ok {
return nil, ErrKeyNotFound
}
next, err := deleteCOWNode(next, string(runeStr[1:]), nodeClone, r)
if err != nil {
return nil, err
}
nodeClone.Next[r] = next
if next == nil {
delete(nodeClone.Next, r)
}
if parent != nil && len(nodeClone.Next) == 0 && nodeClone.Value == "" {
delete(parent.Next, parentKey)
return nil, nil
}
return nodeClone, nil
}