2023-06-20 05:55:12 +00:00
|
|
|
package logstorage
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"unicode"
|
|
|
|
)
|
|
|
|
|
|
|
|
// tokenizeStrings extracts word tokens from a, appends them to dst and returns the result.
|
2024-05-09 18:14:48 +00:00
|
|
|
//
|
|
|
|
// the order of returned tokens is unspecified.
|
2023-06-20 05:55:12 +00:00
|
|
|
func tokenizeStrings(dst, a []string) []string {
|
|
|
|
t := getTokenizer()
|
|
|
|
m := t.m
|
|
|
|
for i, s := range a {
|
|
|
|
if i > 0 && s == a[i-1] {
|
|
|
|
// This string has been already tokenized
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
tokenizeString(m, s)
|
|
|
|
}
|
|
|
|
for k := range t.m {
|
|
|
|
dst = append(dst, k)
|
|
|
|
}
|
|
|
|
putTokenizer(t)
|
|
|
|
|
|
|
|
return dst
|
|
|
|
}
|
|
|
|
|
|
|
|
type tokenizer struct {
|
|
|
|
m map[string]struct{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *tokenizer) reset() {
|
2024-05-09 18:19:01 +00:00
|
|
|
clear(t.m)
|
2023-06-20 05:55:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func tokenizeString(dst map[string]struct{}, s string) {
|
|
|
|
for len(s) > 0 {
|
|
|
|
// Search for the next token.
|
|
|
|
nextIdx := len(s)
|
|
|
|
for i, c := range s {
|
|
|
|
if isTokenRune(c) {
|
|
|
|
nextIdx = i
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s = s[nextIdx:]
|
|
|
|
// Search for the end of the token
|
|
|
|
nextIdx = len(s)
|
|
|
|
for i, c := range s {
|
|
|
|
if !isTokenRune(c) {
|
|
|
|
nextIdx = i
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
token := s[:nextIdx]
|
|
|
|
if len(token) > 0 {
|
|
|
|
dst[token] = struct{}{}
|
|
|
|
}
|
|
|
|
s = s[nextIdx:]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func isTokenRune(c rune) bool {
|
|
|
|
return unicode.IsLetter(c) || unicode.IsDigit(c) || c == '_'
|
|
|
|
}
|
|
|
|
|
|
|
|
func getTokenizer() *tokenizer {
|
|
|
|
v := tokenizerPool.Get()
|
|
|
|
if v == nil {
|
|
|
|
return &tokenizer{
|
|
|
|
m: make(map[string]struct{}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return v.(*tokenizer)
|
|
|
|
}
|
|
|
|
|
|
|
|
func putTokenizer(t *tokenizer) {
|
|
|
|
t.reset()
|
|
|
|
tokenizerPool.Put(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
var tokenizerPool sync.Pool
|
|
|
|
|
|
|
|
type tokensBuf struct {
|
|
|
|
A []string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tb *tokensBuf) reset() {
|
2024-05-09 18:14:48 +00:00
|
|
|
clear(tb.A)
|
|
|
|
tb.A = tb.A[:0]
|
2023-06-20 05:55:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func getTokensBuf() *tokensBuf {
|
|
|
|
v := tokensBufPool.Get()
|
|
|
|
if v == nil {
|
|
|
|
return &tokensBuf{}
|
|
|
|
}
|
|
|
|
return v.(*tokensBuf)
|
|
|
|
}
|
|
|
|
|
|
|
|
func putTokensBuf(tb *tokensBuf) {
|
|
|
|
tb.reset()
|
|
|
|
tokensBufPool.Put(tb)
|
|
|
|
}
|
|
|
|
|
|
|
|
var tokensBufPool sync.Pool
|