|  | // Copyright 2015 The Go Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style | 
|  | // license that can be found in the LICENSE file. | 
|  |  | 
|  | package testing | 
|  |  | 
|  | import ( | 
|  | "fmt" | 
|  | "os" | 
|  | "strconv" | 
|  | "strings" | 
|  | "sync" | 
|  | ) | 
|  |  | 
|  | // matcher sanitizes, uniques, and filters names of subtests and subbenchmarks. | 
|  | type matcher struct { | 
|  | filter    []string | 
|  | matchFunc func(pat, str string) (bool, error) | 
|  |  | 
|  | mu       sync.Mutex | 
|  | subNames map[string]int64 | 
|  | } | 
|  |  | 
|  | // TODO: fix test_main to avoid race and improve caching, also allowing to | 
|  | // eliminate this Mutex. | 
|  | var matchMutex sync.Mutex | 
|  |  | 
|  | func newMatcher(matchString func(pat, str string) (bool, error), patterns, name string) *matcher { | 
|  | var filter []string | 
|  | if patterns != "" { | 
|  | filter = splitRegexp(patterns) | 
|  | for i, s := range filter { | 
|  | filter[i] = rewrite(s) | 
|  | } | 
|  | // Verify filters before doing any processing. | 
|  | for i, s := range filter { | 
|  | if _, err := matchString(s, "non-empty"); err != nil { | 
|  | fmt.Fprintf(os.Stderr, "testing: invalid regexp for element %d of %s (%q): %s\n", i, name, s, err) | 
|  | os.Exit(1) | 
|  | } | 
|  | } | 
|  | } | 
|  | return &matcher{ | 
|  | filter:    filter, | 
|  | matchFunc: matchString, | 
|  | subNames:  map[string]int64{}, | 
|  | } | 
|  | } | 
|  |  | 
|  | func (m *matcher) fullName(c *common, subname string) (name string, ok, partial bool) { | 
|  | name = subname | 
|  |  | 
|  | m.mu.Lock() | 
|  | defer m.mu.Unlock() | 
|  |  | 
|  | if c != nil && c.level > 0 { | 
|  | name = m.unique(c.name, rewrite(subname)) | 
|  | } | 
|  |  | 
|  | matchMutex.Lock() | 
|  | defer matchMutex.Unlock() | 
|  |  | 
|  | // We check the full array of paths each time to allow for the case that | 
|  | // a pattern contains a '/'. | 
|  | elem := strings.Split(name, "/") | 
|  | for i, s := range elem { | 
|  | if i >= len(m.filter) { | 
|  | break | 
|  | } | 
|  | if ok, _ := m.matchFunc(m.filter[i], s); !ok { | 
|  | return name, false, false | 
|  | } | 
|  | } | 
|  | return name, true, len(elem) < len(m.filter) | 
|  | } | 
|  |  | 
|  | func splitRegexp(s string) []string { | 
|  | a := make([]string, 0, strings.Count(s, "/")) | 
|  | cs := 0 | 
|  | cp := 0 | 
|  | for i := 0; i < len(s); { | 
|  | switch s[i] { | 
|  | case '[': | 
|  | cs++ | 
|  | case ']': | 
|  | if cs--; cs < 0 { // An unmatched ']' is legal. | 
|  | cs = 0 | 
|  | } | 
|  | case '(': | 
|  | if cs == 0 { | 
|  | cp++ | 
|  | } | 
|  | case ')': | 
|  | if cs == 0 { | 
|  | cp-- | 
|  | } | 
|  | case '\\': | 
|  | i++ | 
|  | case '/': | 
|  | if cs == 0 && cp == 0 { | 
|  | a = append(a, s[:i]) | 
|  | s = s[i+1:] | 
|  | i = 0 | 
|  | continue | 
|  | } | 
|  | } | 
|  | i++ | 
|  | } | 
|  | return append(a, s) | 
|  | } | 
|  |  | 
|  | // unique creates a unique name for the given parent and subname by affixing it | 
|  | // with one or more counts, if necessary. | 
|  | func (m *matcher) unique(parent, subname string) string { | 
|  | name := fmt.Sprintf("%s/%s", parent, subname) | 
|  | empty := subname == "" | 
|  | for { | 
|  | next, exists := m.subNames[name] | 
|  | if !empty && !exists { | 
|  | m.subNames[name] = 1 // next count is 1 | 
|  | return name | 
|  | } | 
|  | // Name was already used. We increment with the count and append a | 
|  | // string with the count. | 
|  | m.subNames[name] = next + 1 | 
|  |  | 
|  | // Add a count to guarantee uniqueness. | 
|  | name = fmt.Sprintf("%s#%02d", name, next) | 
|  | empty = false | 
|  | } | 
|  | } | 
|  |  | 
|  | // rewrite rewrites a subname to having only printable characters and no white | 
|  | // space. | 
|  | func rewrite(s string) string { | 
|  | b := []byte{} | 
|  | for _, r := range s { | 
|  | switch { | 
|  | case isSpace(r): | 
|  | b = append(b, '_') | 
|  | case !strconv.IsPrint(r): | 
|  | s := strconv.QuoteRune(r) | 
|  | b = append(b, s[1:len(s)-1]...) | 
|  | default: | 
|  | b = append(b, string(r)...) | 
|  | } | 
|  | } | 
|  | return string(b) | 
|  | } | 
|  |  | 
|  | func isSpace(r rune) bool { | 
|  | if r < 0x2000 { | 
|  | switch r { | 
|  | // Note: not the same as Unicode Z class. | 
|  | case '\t', '\n', '\v', '\f', '\r', ' ', 0x85, 0xA0, 0x1680: | 
|  | return true | 
|  | } | 
|  | } else { | 
|  | if r <= 0x200a { | 
|  | return true | 
|  | } | 
|  | switch r { | 
|  | case 0x2028, 0x2029, 0x202f, 0x205f, 0x3000: | 
|  | return true | 
|  | } | 
|  | } | 
|  | return false | 
|  | } |