mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-01 00:30:05 +01:00
148 lines
4.4 KiB
Go
148 lines
4.4 KiB
Go
|
package doublestar
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"os"
|
||
|
"path"
|
||
|
"path/filepath"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
// SplitPattern is a utility function. Given a pattern, SplitPattern will
|
||
|
// return two strings: the first string is everything up to the last slash
|
||
|
// (`/`) that appears _before_ any unescaped "meta" characters (ie, `*?[{`).
|
||
|
// The second string is everything after that slash. For example, given the
|
||
|
// pattern:
|
||
|
//
|
||
|
// ../../path/to/meta*/**
|
||
|
// ^----------- split here
|
||
|
//
|
||
|
// SplitPattern returns "../../path/to" and "meta*/**". This is useful for
|
||
|
// initializing os.DirFS() to call Glob() because Glob() will silently fail if
|
||
|
// your pattern includes `/./` or `/../`. For example:
|
||
|
//
|
||
|
// base, pattern := SplitPattern("../../path/to/meta*/**")
|
||
|
// fsys := os.DirFS(base)
|
||
|
// matches, err := Glob(fsys, pattern)
|
||
|
//
|
||
|
// If SplitPattern cannot find somewhere to split the pattern (for example,
|
||
|
// `meta*/**`), it will return "." and the unaltered pattern (`meta*/**` in
|
||
|
// this example).
|
||
|
//
|
||
|
// Of course, it is your responsibility to decide if the returned base path is
|
||
|
// "safe" in the context of your application. Perhaps you could use Match() to
|
||
|
// validate against a list of approved base directories?
|
||
|
//
|
||
|
func SplitPattern(p string) (base, pattern string) {
|
||
|
base = "."
|
||
|
pattern = p
|
||
|
|
||
|
splitIdx := -1
|
||
|
for i := 0; i < len(p); i++ {
|
||
|
c := p[i]
|
||
|
if c == '\\' {
|
||
|
i++
|
||
|
} else if c == '/' {
|
||
|
splitIdx = i
|
||
|
} else if c == '*' || c == '?' || c == '[' || c == '{' {
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if splitIdx == 0 {
|
||
|
return "/", p[1:]
|
||
|
} else if splitIdx > 0 {
|
||
|
return p[:splitIdx], p[splitIdx+1:]
|
||
|
}
|
||
|
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// FilepathGlob returns the names of all files matching pattern or nil if there
|
||
|
// is no matching file. The syntax of pattern is the same as in Match(). The
|
||
|
// pattern may describe hierarchical names such as usr/*/bin/ed.
|
||
|
//
|
||
|
// FilepathGlob ignores file system errors such as I/O errors reading
|
||
|
// directories by default. The only possible returned error is ErrBadPattern,
|
||
|
// reporting that the pattern is malformed.
|
||
|
//
|
||
|
// To enable aborting on I/O errors, the WithFailOnIOErrors option can be
|
||
|
// passed.
|
||
|
//
|
||
|
// Note: FilepathGlob is a convenience function that is meant as a drop-in
|
||
|
// replacement for `path/filepath.Glob()` for users who don't need the
|
||
|
// complication of io/fs. Basically, it:
|
||
|
// - Runs `filepath.Clean()` and `ToSlash()` on the pattern
|
||
|
// - Runs `SplitPattern()` to get a base path and a pattern to Glob
|
||
|
// - Creates an FS object from the base path and `Glob()s` on the pattern
|
||
|
// - Joins the base path with all of the matches from `Glob()`
|
||
|
//
|
||
|
// Returned paths will use the system's path separator, just like
|
||
|
// `filepath.Glob()`.
|
||
|
//
|
||
|
// Note: the returned error doublestar.ErrBadPattern is not equal to
|
||
|
// filepath.ErrBadPattern.
|
||
|
//
|
||
|
func FilepathGlob(pattern string, opts ...GlobOption) (matches []string, err error) {
|
||
|
pattern = filepath.Clean(pattern)
|
||
|
pattern = filepath.ToSlash(pattern)
|
||
|
base, f := SplitPattern(pattern)
|
||
|
if f == "" || f == "." || f == ".." {
|
||
|
// some special cases to match filepath.Glob behavior
|
||
|
if !ValidatePathPattern(pattern) {
|
||
|
return nil, ErrBadPattern
|
||
|
}
|
||
|
|
||
|
if filepath.Separator != '\\' {
|
||
|
pattern = unescapeMeta(pattern)
|
||
|
}
|
||
|
|
||
|
if _, err = os.Lstat(pattern); err != nil {
|
||
|
g := newGlob(opts...)
|
||
|
if errors.Is(err, os.ErrNotExist) {
|
||
|
return nil, g.handlePatternNotExist(true)
|
||
|
}
|
||
|
return nil, g.forwardErrIfFailOnIOErrors(err)
|
||
|
}
|
||
|
return []string{filepath.FromSlash(pattern)}, nil
|
||
|
}
|
||
|
|
||
|
fs := os.DirFS(base)
|
||
|
if matches, err = Glob(fs, f, opts...); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
for i := range matches {
|
||
|
// use path.Join because we used ToSlash above to ensure our paths are made
|
||
|
// of forward slashes, no matter what the system uses
|
||
|
matches[i] = filepath.FromSlash(path.Join(base, matches[i]))
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Finds the next comma, but ignores any commas that appear inside nested `{}`.
|
||
|
// Assumes that each opening bracket has a corresponding closing bracket.
|
||
|
func indexNextAlt(s string, allowEscaping bool) int {
|
||
|
alts := 1
|
||
|
l := len(s)
|
||
|
for i := 0; i < l; i++ {
|
||
|
if allowEscaping && s[i] == '\\' {
|
||
|
// skip next byte
|
||
|
i++
|
||
|
} else if s[i] == '{' {
|
||
|
alts++
|
||
|
} else if s[i] == '}' {
|
||
|
alts--
|
||
|
} else if s[i] == ',' && alts == 1 {
|
||
|
return i
|
||
|
}
|
||
|
}
|
||
|
return -1
|
||
|
}
|
||
|
|
||
|
var metaReplacer = strings.NewReplacer("\\*", "*", "\\?", "?", "\\[", "[", "\\]", "]", "\\{", "{", "\\}", "}")
|
||
|
|
||
|
// Unescapes meta characters (*?[]{})
|
||
|
func unescapeMeta(pattern string) string {
|
||
|
return metaReplacer.Replace(pattern)
|
||
|
}
|