mirror of https://github.com/gobwas/glob.git
454 lines
11 KiB
Go
454 lines
11 KiB
Go
package glob
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"github.com/gobwas/glob/match"
|
|
"math/rand"
|
|
"regexp"
|
|
"strings"
|
|
"testing"
|
|
)
|
|
|
|
const (
|
|
pattern_all = "[a-z][!a-x]*cat*[h][!b]*eyes*"
|
|
regexp_all = `[a-z][^a-x].*cat.*[h][^b].*eyes.*`
|
|
fixture_all = "my cat has very bright eyes"
|
|
|
|
pattern_plain = "google.com"
|
|
regexp_plain = `google\.com`
|
|
fixture_plain = "google.com"
|
|
|
|
pattern_multiple = "https://*.google.*"
|
|
regexp_multiple = `https:\/\/.*\.google\..*`
|
|
fixture_multiple = "https://account.google.com"
|
|
|
|
pattern_alternatives = "{https://*.google.*,*yandex.*,*yahoo.*,*mail.ru}"
|
|
regexp_alternatives = `(https:\/\/.*\.google\..*|.*yandex\..*|.*yahoo\..*|.*mail\.ru)`
|
|
fixture_alternatives = "http://yahoo.com"
|
|
|
|
pattern_alternatives_suffix = "{https://*gobwas.com,http://exclude.gobwas.com}"
|
|
regexp_alternatives_suffix = `(https:\/\/.*gobwas\.com|http://exclude.gobwas.com)`
|
|
fixture_alternatives_suffix_first = "https://safe.gobwas.com"
|
|
fixture_alternatives_suffix_second = "http://exclude.gobwas.com"
|
|
|
|
pattern_prefix = "abc*"
|
|
regexp_prefix = `abc.*`
|
|
pattern_suffix = "*def"
|
|
regexp_suffix = `.*def`
|
|
pattern_prefix_suffix = "ab*ef"
|
|
regexp_prefix_suffix = `ab.*ef`
|
|
fixture_prefix_suffix = "abcdef"
|
|
|
|
pattern_alternatives_combine_lite = "{abc*def,abc?def,abc[zte]def}"
|
|
regexp_alternatives_combine_lite = `(abc.*def|abc.def|abc[zte]def)`
|
|
fixture_alternatives_combine_lite = "abczdef"
|
|
|
|
pattern_alternatives_combine_hard = "{abc*[a-c]def,abc?[d-g]def,abc[zte]?def}"
|
|
regexp_alternatives_combine_hard = `(abc.*[a-c]def|abc.[d-g]def|abc[zte].def)`
|
|
fixture_alternatives_combine_hard = "abczqdef"
|
|
)
|
|
|
|
type test struct {
|
|
pattern, match string
|
|
should bool
|
|
delimiters []string
|
|
}
|
|
|
|
func glob(s bool, p, m string, d ...string) test {
|
|
return test{p, m, s, d}
|
|
}
|
|
|
|
func draw(pattern string, m match.Matcher) string {
|
|
return fmt.Sprintf(`digraph G {graph[label="%s"];%s}`, pattern, graphviz(m, fmt.Sprintf("%x", rand.Int63())))
|
|
}
|
|
|
|
func graphviz(m match.Matcher, id string) string {
|
|
buf := &bytes.Buffer{}
|
|
|
|
switch matcher := m.(type) {
|
|
case match.BTree:
|
|
fmt.Fprintf(buf, `"%s"[label="%s"];`, id, matcher.Value.String())
|
|
for _, m := range []match.Matcher{matcher.Left, matcher.Right} {
|
|
switch n := m.(type) {
|
|
case nil:
|
|
rnd := rand.Int63()
|
|
fmt.Fprintf(buf, `"%x"[label="<nil>"];`, rnd)
|
|
fmt.Fprintf(buf, `"%s"->"%x";`, id, rnd)
|
|
|
|
default:
|
|
sub := fmt.Sprintf("%x", rand.Int63())
|
|
fmt.Fprintf(buf, `"%s"->"%s";`, id, sub)
|
|
fmt.Fprintf(buf, graphviz(n, sub))
|
|
}
|
|
}
|
|
|
|
case match.AnyOf:
|
|
fmt.Fprintf(buf, `"%s"[label="AnyOf"];`, id)
|
|
for _, m := range matcher.Matchers {
|
|
rnd := rand.Int63()
|
|
fmt.Fprintf(buf, graphviz(m, fmt.Sprintf("%x", rnd)))
|
|
fmt.Fprintf(buf, `"%s"->"%x";`, id, rnd)
|
|
}
|
|
|
|
case match.EveryOf:
|
|
fmt.Fprintf(buf, `"%s"[label="EveryOf"];`, id)
|
|
for _, m := range matcher.Matchers {
|
|
rnd := rand.Int63()
|
|
fmt.Fprintf(buf, graphviz(m, fmt.Sprintf("%x", rnd)))
|
|
fmt.Fprintf(buf, `"%s"->"%x";`, id, rnd)
|
|
}
|
|
|
|
default:
|
|
fmt.Fprintf(buf, `"%s"[label="%s"];`, id, m.String())
|
|
}
|
|
|
|
return buf.String()
|
|
}
|
|
|
|
func DrawPatterns(t *testing.T) {
|
|
for id, test := range []struct {
|
|
pattern string
|
|
sep string
|
|
}{
|
|
{
|
|
pattern: pattern_all,
|
|
},
|
|
{
|
|
pattern: pattern_alternatives_suffix,
|
|
sep: separators,
|
|
},
|
|
{
|
|
pattern: pattern_alternatives_combine_lite,
|
|
},
|
|
{
|
|
pattern: pattern_alternatives_combine_hard,
|
|
},
|
|
{
|
|
pattern: "{https://*.mail.ru,*my.mail.ru,*my.myalpha*.i.mail.ru}",
|
|
},
|
|
} {
|
|
glob, err := Compile(test.pattern, test.sep)
|
|
if err != nil {
|
|
t.Errorf("#%d compile pattern error: %s", id, err)
|
|
continue
|
|
}
|
|
|
|
matcher := glob.(match.Matcher)
|
|
fmt.Println(test.pattern)
|
|
fmt.Println(strings.Repeat("=", len(test.pattern)))
|
|
fmt.Println(draw(test.pattern, matcher))
|
|
fmt.Println()
|
|
fmt.Println(matcher.String())
|
|
fmt.Println()
|
|
}
|
|
}
|
|
|
|
func TestIndexByteNonEscaped(t *testing.T) {
|
|
for _, test := range []struct {
|
|
s string
|
|
n, e byte
|
|
i int
|
|
}{
|
|
{
|
|
"\\n_n",
|
|
'n',
|
|
'\\',
|
|
3,
|
|
},
|
|
{
|
|
"ab",
|
|
'a',
|
|
'\\',
|
|
0,
|
|
},
|
|
{
|
|
"ab",
|
|
'b',
|
|
'\\',
|
|
1,
|
|
},
|
|
{
|
|
"",
|
|
'b',
|
|
'\\',
|
|
-1,
|
|
},
|
|
{
|
|
"\\b",
|
|
'b',
|
|
'\\',
|
|
-1,
|
|
},
|
|
} {
|
|
i := indexByteNonEscaped(test.s, test.n, test.e, 0)
|
|
if i != test.i {
|
|
t.Errorf("unexpeted index: expected %v, got %v", test.i, i)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestGlob(t *testing.T) {
|
|
for _, test := range []test{
|
|
glob(true, "* ?at * eyes", "my cat has very bright eyes"),
|
|
|
|
glob(true, "abc", "abc"),
|
|
glob(true, "a*c", "abc"),
|
|
glob(true, "a*c", "a12345c"),
|
|
glob(true, "a?c", "a1c"),
|
|
glob(true, "a.b", "a.b", "."),
|
|
glob(true, "a.*", "a.b", "."),
|
|
glob(true, "a.**", "a.b.c", "."),
|
|
glob(true, "a.?.c", "a.b.c", "."),
|
|
glob(true, "a.?.?", "a.b.c", "."),
|
|
glob(true, "?at", "cat"),
|
|
glob(true, "?at", "fat"),
|
|
glob(true, "*", "abc"),
|
|
glob(true, `\*`, "*"),
|
|
glob(true, "**", "a.b.c", "."),
|
|
|
|
glob(false, "?at", "at"),
|
|
glob(false, "?at", "fat", "f"),
|
|
glob(false, "a.*", "a.b.c", "."),
|
|
glob(false, "a.?.c", "a.bb.c", "."),
|
|
glob(false, "*", "a.b.c", "."),
|
|
|
|
glob(true, "*test", "this is a test"),
|
|
glob(true, "this*", "this is a test"),
|
|
glob(true, "*is *", "this is a test"),
|
|
glob(true, "*is*a*", "this is a test"),
|
|
glob(true, "**test**", "this is a test"),
|
|
glob(true, "**is**a***test*", "this is a test"),
|
|
|
|
glob(false, "*is", "this is a test"),
|
|
glob(false, "*no*", "this is a test"),
|
|
glob(true, "[!a]*", "this is a test3"),
|
|
|
|
glob(true, "*abc", "abcabc"),
|
|
glob(true, "**abc", "abcabc"),
|
|
glob(true, "???", "abc"),
|
|
glob(true, "?*?", "abc"),
|
|
glob(true, "?*?", "ac"),
|
|
|
|
glob(true, "{abc,def}ghi", "defghi"),
|
|
glob(true, "{abc,abcd}a", "abcda"),
|
|
glob(true, "{a,ab}{bc,f}", "abc"),
|
|
glob(true, "{*,**}{a,b}", "ab"),
|
|
glob(false, "{*,**}{a,b}", "ac"),
|
|
|
|
glob(true, pattern_all, fixture_all),
|
|
glob(true, pattern_plain, fixture_plain),
|
|
glob(true, pattern_multiple, fixture_multiple),
|
|
glob(true, pattern_alternatives, fixture_alternatives),
|
|
glob(true, pattern_alternatives_suffix, fixture_alternatives_suffix_first),
|
|
glob(true, pattern_alternatives_suffix, fixture_alternatives_suffix_second),
|
|
glob(true, pattern_alternatives_combine_hard, fixture_alternatives_combine_hard),
|
|
glob(true, pattern_alternatives_combine_lite, fixture_alternatives_combine_lite),
|
|
glob(true, pattern_prefix, fixture_prefix_suffix),
|
|
glob(true, pattern_suffix, fixture_prefix_suffix),
|
|
glob(true, pattern_prefix_suffix, fixture_prefix_suffix),
|
|
} {
|
|
g, err := Compile(test.pattern, test.delimiters...)
|
|
if err != nil {
|
|
t.Errorf("parsing pattern %q error: %s", test.pattern, err)
|
|
continue
|
|
}
|
|
|
|
result := g.Match(test.match)
|
|
if result != test.should {
|
|
t.Errorf("pattern %q matching %q should be %v but got %v\n%s", test.pattern, test.match, test.should, result, g)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkParse(b *testing.B) {
|
|
for i := 0; i < b.N; i++ {
|
|
Compile(pattern_all)
|
|
}
|
|
}
|
|
func BenchmarkParseRegexp(b *testing.B) {
|
|
for i := 0; i < b.N; i++ {
|
|
regexp.MustCompile(regexp_all)
|
|
}
|
|
}
|
|
|
|
func BenchmarkAll(b *testing.B) {
|
|
m, _ := Compile(pattern_all)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_all)
|
|
}
|
|
}
|
|
func BenchmarkAllRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_all)
|
|
f := []byte(fixture_all)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
|
|
func BenchmarkMultiple(b *testing.B) {
|
|
m, _ := Compile(pattern_multiple)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_multiple)
|
|
}
|
|
}
|
|
func BenchmarkMultipleRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_multiple)
|
|
f := []byte(fixture_multiple)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
|
|
func BenchmarkAlternatives(b *testing.B) {
|
|
m, _ := Compile(pattern_alternatives)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_alternatives)
|
|
}
|
|
}
|
|
func BenchmarkAlternativesSuffixFirst(b *testing.B) {
|
|
m, _ := Compile(pattern_alternatives_suffix)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_alternatives_suffix_first)
|
|
}
|
|
}
|
|
func BenchmarkAlternativesSuffixSecond(b *testing.B) {
|
|
m, _ := Compile(pattern_alternatives_suffix)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_alternatives_suffix_second)
|
|
}
|
|
}
|
|
func BenchmarkAlternativesCombineLite(b *testing.B) {
|
|
m, _ := Compile(pattern_alternatives_combine_lite)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_alternatives_combine_lite)
|
|
}
|
|
}
|
|
func BenchmarkAlternativesCombineHard(b *testing.B) {
|
|
m, _ := Compile(pattern_alternatives_combine_hard)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_alternatives_combine_hard)
|
|
}
|
|
}
|
|
func BenchmarkAlternativesRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_alternatives)
|
|
f := []byte(fixture_alternatives)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
func BenchmarkAlternativesSuffixFirstRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_alternatives_suffix)
|
|
f := []byte(fixture_alternatives_suffix_first)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
func BenchmarkAlternativesSuffixSecondRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_alternatives_suffix)
|
|
f := []byte(fixture_alternatives_suffix_second)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
func BenchmarkAlternativesCombineLiteRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_alternatives_combine_lite)
|
|
f := []byte(fixture_alternatives_combine_lite)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
func BenchmarkAlternativesCombineHardRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_alternatives_combine_hard)
|
|
f := []byte(fixture_alternatives_combine_hard)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
|
|
func BenchmarkPlain(b *testing.B) {
|
|
m, _ := Compile(pattern_plain)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_plain)
|
|
}
|
|
}
|
|
func BenchmarkPlainRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_plain)
|
|
f := []byte(fixture_plain)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
|
|
func BenchmarkPrefix(b *testing.B) {
|
|
m, _ := Compile(pattern_prefix)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_prefix_suffix)
|
|
}
|
|
}
|
|
func BenchmarkPrefixRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_prefix)
|
|
f := []byte(fixture_prefix_suffix)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
|
|
func BenchmarkSuffix(b *testing.B) {
|
|
m, _ := Compile(pattern_suffix)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_prefix_suffix)
|
|
}
|
|
}
|
|
func BenchmarkSuffixRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_suffix)
|
|
f := []byte(fixture_prefix_suffix)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
|
|
func BenchmarkPrefixSuffix(b *testing.B) {
|
|
m, _ := Compile(pattern_prefix_suffix)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(fixture_prefix_suffix)
|
|
}
|
|
}
|
|
func BenchmarkPrefixSuffixRegexp(b *testing.B) {
|
|
m := regexp.MustCompile(regexp_prefix_suffix)
|
|
f := []byte(fixture_prefix_suffix)
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
_ = m.Match(f)
|
|
}
|
|
}
|
|
|
|
//BenchmarkParse-8 500000 2235 ns/op
|
|
//BenchmarkAll-8 20000000 73.1 ns/op
|
|
//BenchmarkMultiple-8 10000000 130 ns/op
|
|
//BenchmarkPlain-8 200000000 6.70 ns/op
|
|
//BenchmarkPrefix-8 200000000 8.36 ns/op
|
|
//BenchmarkSuffix-8 200000000 8.35 ns/op
|
|
//BenchmarkPrefixSuffix-8 100000000 13.6 ns/op
|