-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnfa.go
81 lines (69 loc) · 1.78 KB
/
nfa.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
package hypermatch
import "slices"
type nfaStep struct {
ValueTransitions map[byte]*nfaStep `json:"v,omitempty"`
FieldTransition []*fieldMatcher `json:"f,omitempty"`
}
func newNfaStep() *nfaStep {
return &nfaStep{
ValueTransitions: make(map[byte]*nfaStep),
FieldTransition: nil,
}
}
func (n *nfaStep) MakeStep(char byte) *nfaStep {
s, ok := n.ValueTransitions[char]
if !ok {
s = newNfaStep()
n.ValueTransitions[char] = s
}
return s
}
func (n *nfaStep) addOrReuseOrCreateFieldTransition(fm *fieldMatcher) *fieldMatcher {
if fm != nil {
if slices.Index(n.FieldTransition, fm) == -1 {
n.FieldTransition = append(n.FieldTransition, fm)
return fm
} else {
return fm
}
} else {
for _, f := range n.FieldTransition {
if !f.Exclusive {
return f
}
}
fm := newFieldMatcher()
n.FieldTransition = append(n.FieldTransition, fm)
return fm
}
}
func transitionNfa(step *nfaStep, value []byte, transitions []*fieldMatcher) []*fieldMatcher {
if len(value) == 0 {
return nil
}
if step.FieldTransition != nil {
transitions = append(transitions, step.FieldTransition...)
}
// transition through the NFA
for i, v := range value {
// if there are no value transitions we can stop here
if step.ValueTransitions == nil {
break
}
// if there is a wildcard transition, run through it recursively
if w, ok := step.ValueTransitions[byteWildcard]; ok && len(value) > i+1 {
transitions = append(transitions, transitionNfa(w, value[i+1:], nil)...)
}
// follow the next step if possible, otherwise stop
if s, ok := step.ValueTransitions[v]; ok {
step = s
} else {
break
}
// if there are field transitions, add them!
if step.FieldTransition != nil {
transitions = append(transitions, step.FieldTransition...)
}
}
return transitions
}