123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185 |
- package match
- import (
- "fmt"
- "unicode/utf8"
- )
- type BTree struct {
- Value Matcher
- Left Matcher
- Right Matcher
- ValueLengthRunes int
- LeftLengthRunes int
- RightLengthRunes int
- LengthRunes int
- }
- func NewBTree(Value, Left, Right Matcher) (tree BTree) {
- tree.Value = Value
- tree.Left = Left
- tree.Right = Right
- lenOk := true
- if tree.ValueLengthRunes = Value.Len(); tree.ValueLengthRunes == -1 {
- lenOk = false
- }
- if Left != nil {
- if tree.LeftLengthRunes = Left.Len(); tree.LeftLengthRunes == -1 {
- lenOk = false
- }
- }
- if Right != nil {
- if tree.RightLengthRunes = Right.Len(); tree.RightLengthRunes == -1 {
- lenOk = false
- }
- }
- if lenOk {
- tree.LengthRunes = tree.LeftLengthRunes + tree.ValueLengthRunes + tree.RightLengthRunes
- } else {
- tree.LengthRunes = -1
- }
- return tree
- }
- func (self BTree) Len() int {
- return self.LengthRunes
- }
- // todo?
- func (self BTree) Index(s string) (index int, segments []int) {
- //inputLen := len(s)
- //// try to cut unnecessary parts
- //// by knowledge of length of right and left part
- //offset, limit := self.offsetLimit(inputLen)
- //for offset < limit {
- // // search for matching part in substring
- // vi, segments := self.Value.Index(s[offset:limit])
- // if index == -1 {
- // return -1, nil
- // }
- // if self.Left == nil {
- // if index != offset {
- // return -1, nil
- // }
- // } else {
- // left := s[:offset+vi]
- // i := self.Left.IndexSuffix(left)
- // if i == -1 {
- // return -1, nil
- // }
- // index = i
- // }
- // if self.Right != nil {
- // for _, seg := range segments {
- // right := s[:offset+vi+seg]
- // }
- // }
- // l := s[:offset+index]
- // var left bool
- // if self.Left != nil {
- // left = self.Left.Index(l)
- // } else {
- // left = l == ""
- // }
- //}
- return -1, nil
- }
- func (self BTree) Match(s string) bool {
- inputLen := len(s)
- // try to cut unnecessary parts
- // by knowledge of length of right and left part
- offset, limit := self.offsetLimit(inputLen)
- for offset < limit {
- // search for matching part in substring
- index, segments := self.Value.Index(s[offset:limit])
- if index == -1 {
- releaseSegments(segments)
- return false
- }
- l := s[:offset+index]
- var left bool
- if self.Left != nil {
- left = self.Left.Match(l)
- } else {
- left = l == ""
- }
- if left {
- for i := len(segments) - 1; i >= 0; i-- {
- length := segments[i]
- var right bool
- var r string
- // if there is no string for the right branch
- if inputLen <= offset+index+length {
- r = ""
- } else {
- r = s[offset+index+length:]
- }
- if self.Right != nil {
- right = self.Right.Match(r)
- } else {
- right = r == ""
- }
- if right {
- releaseSegments(segments)
- return true
- }
- }
- }
- _, step := utf8.DecodeRuneInString(s[offset+index:])
- offset += index + step
- releaseSegments(segments)
- }
- return false
- }
- func (self BTree) offsetLimit(inputLen int) (offset int, limit int) {
- // self.Length, self.RLen and self.LLen are values meaning the length of runes for each part
- // here we manipulating byte length for better optimizations
- // but these checks still works, cause minLen of 1-rune string is 1 byte.
- if self.LengthRunes != -1 && self.LengthRunes > inputLen {
- return 0, 0
- }
- if self.LeftLengthRunes >= 0 {
- offset = self.LeftLengthRunes
- }
- if self.RightLengthRunes >= 0 {
- limit = inputLen - self.RightLengthRunes
- } else {
- limit = inputLen
- }
- return offset, limit
- }
- func (self BTree) String() string {
- const n string = "<nil>"
- var l, r string
- if self.Left == nil {
- l = n
- } else {
- l = self.Left.String()
- }
- if self.Right == nil {
- r = n
- } else {
- r = self.Right.String()
- }
- return fmt.Sprintf("<btree:[%s<-%s->%s]>", l, self.Value, r)
- }
|