2020-11-22 06:02:27 -08:00
|
|
|
package main
|
|
|
|
|
2020-12-30 13:01:39 -08:00
|
|
|
import (
|
2021-10-20 08:01:19 -07:00
|
|
|
"fmt"
|
2020-12-30 13:01:39 -08:00
|
|
|
"regexp"
|
2021-10-20 08:01:19 -07:00
|
|
|
"strings"
|
2020-12-30 13:01:39 -08:00
|
|
|
"sync"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2021-05-21 11:01:08 -07:00
|
|
|
regexCache = make(map[string]*regexp.Regexp)
|
|
|
|
regexCacheLock = sync.RWMutex{}
|
2020-12-30 13:01:39 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
func getCompiledRegex(pattern string) *regexp.Regexp {
|
|
|
|
// try in cache first
|
2021-02-28 21:11:35 -08:00
|
|
|
regexCacheLock.RLock()
|
2020-12-30 13:01:39 -08:00
|
|
|
re := regexCache[pattern]
|
2021-02-28 21:11:35 -08:00
|
|
|
regexCacheLock.RUnlock()
|
2020-12-30 13:01:39 -08:00
|
|
|
if re != nil {
|
|
|
|
return re
|
|
|
|
}
|
|
|
|
|
|
|
|
// should we panic or return the error?
|
|
|
|
re = regexp.MustCompile(pattern)
|
|
|
|
|
|
|
|
// lock for concurrent access and save the compiled expression in cache
|
|
|
|
regexCacheLock.Lock()
|
|
|
|
regexCache[pattern] = re
|
|
|
|
regexCacheLock.Unlock()
|
|
|
|
|
|
|
|
return re
|
|
|
|
}
|
2020-11-22 06:02:27 -08:00
|
|
|
|
|
|
|
func findNamedRegexMatch(pattern, text string) map[string]string {
|
2020-12-30 13:01:39 -08:00
|
|
|
// error ignored because mustCompile will cause a panic
|
|
|
|
re := getCompiledRegex(pattern)
|
2020-11-22 06:02:27 -08:00
|
|
|
match := re.FindStringSubmatch(text)
|
|
|
|
result := make(map[string]string)
|
|
|
|
if len(match) == 0 {
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
for i, name := range re.SubexpNames() {
|
|
|
|
if i == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
result[name] = match[i]
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func findAllNamedRegexMatch(pattern, text string) []map[string]string {
|
2020-12-30 13:01:39 -08:00
|
|
|
re := getCompiledRegex(pattern)
|
2020-11-22 06:02:27 -08:00
|
|
|
match := re.FindAllStringSubmatch(text, -1)
|
|
|
|
var results []map[string]string
|
|
|
|
if len(match) == 0 {
|
|
|
|
return results
|
|
|
|
}
|
|
|
|
for _, set := range match {
|
|
|
|
result := make(map[string]string)
|
|
|
|
for i, name := range re.SubexpNames() {
|
|
|
|
if i == 0 {
|
|
|
|
result["text"] = set[i]
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
result[name] = set[i]
|
|
|
|
}
|
|
|
|
results = append(results, result)
|
|
|
|
}
|
|
|
|
return results
|
|
|
|
}
|
|
|
|
|
|
|
|
func replaceAllString(pattern, text, replaceText string) string {
|
2020-12-30 13:01:39 -08:00
|
|
|
re := getCompiledRegex(pattern)
|
2020-11-22 06:02:27 -08:00
|
|
|
return re.ReplaceAllString(text, replaceText)
|
|
|
|
}
|
2020-12-30 13:01:39 -08:00
|
|
|
|
|
|
|
func matchString(pattern, text string) bool {
|
|
|
|
re := getCompiledRegex(pattern)
|
|
|
|
return re.MatchString(text)
|
|
|
|
}
|
2021-10-20 08:01:19 -07:00
|
|
|
|
|
|
|
func dirMatchesOneOf(env environmentInfo, dir string, regexes []string) bool {
|
|
|
|
normalizedCwd := strings.ReplaceAll(dir, "\\", "/")
|
|
|
|
normalizedHomeDir := strings.ReplaceAll(env.homeDir(), "\\", "/")
|
|
|
|
|
|
|
|
for _, element := range regexes {
|
|
|
|
normalizedElement := strings.ReplaceAll(element, "\\\\", "/")
|
|
|
|
if strings.HasPrefix(normalizedElement, "~") {
|
|
|
|
normalizedElement = normalizedHomeDir + normalizedElement[1:]
|
|
|
|
}
|
|
|
|
pattern := fmt.Sprintf("^%s$", normalizedElement)
|
|
|
|
goos := env.getRuntimeGOOS()
|
|
|
|
if goos == windowsPlatform || goos == darwinPlatform {
|
|
|
|
pattern = "(?i)" + pattern
|
|
|
|
}
|
|
|
|
matched := matchString(pattern, normalizedCwd)
|
|
|
|
if matched {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|