-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnode.go
97 lines (82 loc) · 1.7 KB
/
node.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
package arn
// NewNode returns a new node using txt as level value. E.g. +, # or a
// word.
func NewNode(txt string) *Node {
return &Node{
txt: txt,
}
}
type Node struct {
// Value is controlled by the caller.
Value any
txt string
parent *Node
children []*Node
}
func (n *Node) match(result *[]*Node, parts []string, i int) {
lastPart := len(parts)-1 == i
switch {
case i > len(parts)-1:
*result = append(*result, n)
return
case n.txt == "#":
if len(parts[0]) == 0 || parts[0][0] != '$' {
// https://docs.oasis-open.org/mqtt/mqtt/v5.0/os/mqtt-v5.0-os.html#_Toc3901246
*result = append(*result, n)
}
return
case lastPart && n.txt == "+":
*result = append(*result, n)
return
case lastPart && n.txt == parts[i] && n.IsLeaf():
*result = append(*result, n)
return
case n.txt != "+" && n.txt != parts[i]:
return
}
for _, child := range n.children {
child.match(result, parts, i+1)
}
}
func (n *Node) Find(parts []string) (*Node, bool) {
if len(parts) == 0 {
return n, true
}
c := n.FindChild(parts[0])
if c == nil {
return nil, false
}
return c.Find(parts[1:])
}
func (n *Node) FindChild(txt string) *Node {
for _, child := range n.children {
if child.txt == txt {
return child
}
}
return nil
}
func (n *Node) AddChild(c *Node) {
c.parent = n
n.children = append(n.children, c)
}
func (n *Node) Filter() string {
if n.parent == nil {
return n.txt
}
return n.parent.Filter() + "/" + n.txt
}
func (n *Node) Leafs() []*Node {
var leafs []*Node
for _, c := range n.children {
if c.IsLeaf() {
leafs = append(leafs, c)
continue
}
leafs = append(leafs, c.Leafs()...)
}
return leafs
}
func (n *Node) IsLeaf() bool {
return len(n.children) == 0
}