Merge pull request #2311 from vincentbernat/feature/negative-pattern
filter: ability to use negative patterns
This commit is contained in:
commit
3a285f91bc
4 changed files with 139 additions and 45 deletions
31
changelog/unreleased/issue-233
Normal file
31
changelog/unreleased/issue-233
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
Enhancement: Add negative patterns for include/exclude
|
||||||
|
|
||||||
|
If a pattern starts with an exclamation mark and it matches a file that
|
||||||
|
was previously matched by a regular pattern, the match is cancelled.
|
||||||
|
Notably, this can be used with `--exclude-file` to cancel the
|
||||||
|
exclusion of some files.
|
||||||
|
|
||||||
|
It works similarly to `gitignore`, with the same limitation: once a
|
||||||
|
directory is excluded, it is not possible to include files inside the
|
||||||
|
directory.
|
||||||
|
|
||||||
|
Example of use (as an exclude pattern for backup):
|
||||||
|
|
||||||
|
$HOME/**/*
|
||||||
|
!$HOME/Documents
|
||||||
|
!$HOME/code
|
||||||
|
!$HOME/.emacs.d
|
||||||
|
!$HOME/games
|
||||||
|
# [...]
|
||||||
|
node_modules
|
||||||
|
*~
|
||||||
|
*.o
|
||||||
|
*.lo
|
||||||
|
*.pyc
|
||||||
|
# [...]
|
||||||
|
$HOME/code/linux/*
|
||||||
|
!$HOME/code/linux/.git
|
||||||
|
# [...]
|
||||||
|
|
||||||
|
https://github.com/restic/restic/issues/233
|
||||||
|
https://github.com/restic/restic/pull/2311
|
|
@ -289,6 +289,28 @@ On most Unixy shells, you can either quote or use backslashes. For example:
|
||||||
* ``--exclude="foo bar star/foo.txt"``
|
* ``--exclude="foo bar star/foo.txt"``
|
||||||
* ``--exclude=foo\ bar\ star/foo.txt``
|
* ``--exclude=foo\ bar\ star/foo.txt``
|
||||||
|
|
||||||
|
If a pattern starts with exclamation mark and matches a file that
|
||||||
|
was previously matched by a regular pattern, the match is cancelled.
|
||||||
|
It works similarly to ``gitignore``, with the same limitation: once a
|
||||||
|
directory is excluded, it is not possible to include files inside the
|
||||||
|
directory. Here is a complete example to backup a selection of
|
||||||
|
directories inside the home directory. It works by excluding any
|
||||||
|
directory, then selectively add back some of them.
|
||||||
|
|
||||||
|
::
|
||||||
|
|
||||||
|
$HOME/**/*
|
||||||
|
!$HOME/Documents
|
||||||
|
!$HOME/code
|
||||||
|
!$HOME/.emacs.d
|
||||||
|
!$HOME/games
|
||||||
|
# [...]
|
||||||
|
node_modules
|
||||||
|
*~
|
||||||
|
*.o
|
||||||
|
*.lo
|
||||||
|
*.pyc
|
||||||
|
|
||||||
By specifying the option ``--one-file-system`` you can instruct restic
|
By specifying the option ``--one-file-system`` you can instruct restic
|
||||||
to only backup files from the file systems the initially specified files
|
to only backup files from the file systems the initially specified files
|
||||||
or directories reside on. In other words, it will prevent restic from crossing
|
or directories reside on. In other words, it will prevent restic from crossing
|
||||||
|
|
|
@ -17,7 +17,10 @@ type patternPart struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Pattern represents a preparsed filter pattern
|
// Pattern represents a preparsed filter pattern
|
||||||
type Pattern []patternPart
|
type Pattern struct {
|
||||||
|
parts []patternPart
|
||||||
|
isNegated bool
|
||||||
|
}
|
||||||
|
|
||||||
func prepareStr(str string) ([]string, error) {
|
func prepareStr(str string) ([]string, error) {
|
||||||
if str == "" {
|
if str == "" {
|
||||||
|
@ -26,20 +29,26 @@ func prepareStr(str string) ([]string, error) {
|
||||||
return splitPath(str), nil
|
return splitPath(str), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func preparePattern(pattern string) Pattern {
|
func preparePattern(patternStr string) Pattern {
|
||||||
parts := splitPath(filepath.Clean(pattern))
|
var negate bool
|
||||||
patterns := make([]patternPart, len(parts))
|
if patternStr[0] == '!' {
|
||||||
for i, part := range parts {
|
negate = true
|
||||||
|
patternStr = patternStr[1:]
|
||||||
|
}
|
||||||
|
|
||||||
|
pathParts := splitPath(filepath.Clean(patternStr))
|
||||||
|
parts := make([]patternPart, len(pathParts))
|
||||||
|
for i, part := range pathParts {
|
||||||
isSimple := !strings.ContainsAny(part, "\\[]*?")
|
isSimple := !strings.ContainsAny(part, "\\[]*?")
|
||||||
// Replace "**" with the empty string to get faster comparisons
|
// Replace "**" with the empty string to get faster comparisons
|
||||||
// (length-check only) in hasDoubleWildcard.
|
// (length-check only) in hasDoubleWildcard.
|
||||||
if part == "**" {
|
if part == "**" {
|
||||||
part = ""
|
part = ""
|
||||||
}
|
}
|
||||||
patterns[i] = patternPart{part, isSimple}
|
parts[i] = patternPart{part, isSimple}
|
||||||
}
|
}
|
||||||
|
|
||||||
return patterns
|
return Pattern{parts, negate}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Split p into path components. Assuming p has been Cleaned, no component
|
// Split p into path components. Assuming p has been Cleaned, no component
|
||||||
|
@ -62,19 +71,19 @@ func splitPath(p string) []string {
|
||||||
// In addition patterns suitable for filepath.Match, pattern accepts a
|
// In addition patterns suitable for filepath.Match, pattern accepts a
|
||||||
// recursive wildcard '**', which greedily matches an arbitrary number of
|
// recursive wildcard '**', which greedily matches an arbitrary number of
|
||||||
// intermediate directories.
|
// intermediate directories.
|
||||||
func Match(pattern, str string) (matched bool, err error) {
|
func Match(patternStr, str string) (matched bool, err error) {
|
||||||
if pattern == "" {
|
if patternStr == "" {
|
||||||
return true, nil
|
return true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
patterns := preparePattern(pattern)
|
pattern := preparePattern(patternStr)
|
||||||
strs, err := prepareStr(str)
|
strs, err := prepareStr(str)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
return match(patterns, strs)
|
return match(pattern, strs)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ChildMatch returns true if children of str can match the pattern. When the pattern is
|
// ChildMatch returns true if children of str can match the pattern. When the pattern is
|
||||||
|
@ -87,28 +96,28 @@ func Match(pattern, str string) (matched bool, err error) {
|
||||||
// In addition patterns suitable for filepath.Match, pattern accepts a
|
// In addition patterns suitable for filepath.Match, pattern accepts a
|
||||||
// recursive wildcard '**', which greedily matches an arbitrary number of
|
// recursive wildcard '**', which greedily matches an arbitrary number of
|
||||||
// intermediate directories.
|
// intermediate directories.
|
||||||
func ChildMatch(pattern, str string) (matched bool, err error) {
|
func ChildMatch(patternStr, str string) (matched bool, err error) {
|
||||||
if pattern == "" {
|
if patternStr == "" {
|
||||||
return true, nil
|
return true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
patterns := preparePattern(pattern)
|
pattern := preparePattern(patternStr)
|
||||||
strs, err := prepareStr(str)
|
strs, err := prepareStr(str)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
return childMatch(patterns, strs)
|
return childMatch(pattern, strs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func childMatch(patterns Pattern, strs []string) (matched bool, err error) {
|
func childMatch(pattern Pattern, strs []string) (matched bool, err error) {
|
||||||
if patterns[0].pattern != "/" {
|
if pattern.parts[0].pattern != "/" {
|
||||||
// relative pattern can always be nested down
|
// relative pattern can always be nested down
|
||||||
return true, nil
|
return true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
ok, pos := hasDoubleWildcard(patterns)
|
ok, pos := hasDoubleWildcard(pattern)
|
||||||
if ok && len(strs) >= pos {
|
if ok && len(strs) >= pos {
|
||||||
// cut off at the double wildcard
|
// cut off at the double wildcard
|
||||||
strs = strs[:pos]
|
strs = strs[:pos]
|
||||||
|
@ -116,16 +125,16 @@ func childMatch(patterns Pattern, strs []string) (matched bool, err error) {
|
||||||
|
|
||||||
// match path against absolute pattern prefix
|
// match path against absolute pattern prefix
|
||||||
l := 0
|
l := 0
|
||||||
if len(strs) > len(patterns) {
|
if len(strs) > len(pattern.parts) {
|
||||||
l = len(patterns)
|
l = len(pattern.parts)
|
||||||
} else {
|
} else {
|
||||||
l = len(strs)
|
l = len(strs)
|
||||||
}
|
}
|
||||||
return match(patterns[0:l], strs)
|
return match(Pattern{pattern.parts[0:l], pattern.isNegated}, strs)
|
||||||
}
|
}
|
||||||
|
|
||||||
func hasDoubleWildcard(list Pattern) (ok bool, pos int) {
|
func hasDoubleWildcard(list Pattern) (ok bool, pos int) {
|
||||||
for i, item := range list {
|
for i, item := range list.parts {
|
||||||
if item.pattern == "" {
|
if item.pattern == "" {
|
||||||
return true, i
|
return true, i
|
||||||
}
|
}
|
||||||
|
@ -134,22 +143,22 @@ func hasDoubleWildcard(list Pattern) (ok bool, pos int) {
|
||||||
return false, 0
|
return false, 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func match(patterns Pattern, strs []string) (matched bool, err error) {
|
func match(pattern Pattern, strs []string) (matched bool, err error) {
|
||||||
if ok, pos := hasDoubleWildcard(patterns); ok {
|
if ok, pos := hasDoubleWildcard(pattern); ok {
|
||||||
// gradually expand '**' into separate wildcards
|
// gradually expand '**' into separate wildcards
|
||||||
newPat := make(Pattern, len(strs))
|
newPat := make([]patternPart, len(strs))
|
||||||
// copy static prefix once
|
// copy static prefix once
|
||||||
copy(newPat, patterns[:pos])
|
copy(newPat, pattern.parts[:pos])
|
||||||
for i := 0; i <= len(strs)-len(patterns)+1; i++ {
|
for i := 0; i <= len(strs)-len(pattern.parts)+1; i++ {
|
||||||
// limit to static prefix and already appended '*'
|
// limit to static prefix and already appended '*'
|
||||||
newPat := newPat[:pos+i]
|
newPat := newPat[:pos+i]
|
||||||
// in the first iteration the wildcard expands to nothing
|
// in the first iteration the wildcard expands to nothing
|
||||||
if i > 0 {
|
if i > 0 {
|
||||||
newPat[pos+i-1] = patternPart{"*", false}
|
newPat[pos+i-1] = patternPart{"*", false}
|
||||||
}
|
}
|
||||||
newPat = append(newPat, patterns[pos+1:]...)
|
newPat = append(newPat, pattern.parts[pos+1:]...)
|
||||||
|
|
||||||
matched, err := match(newPat, strs)
|
matched, err := match(Pattern{newPat, pattern.isNegated}, strs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, err
|
return false, err
|
||||||
}
|
}
|
||||||
|
@ -162,20 +171,20 @@ func match(patterns Pattern, strs []string) (matched bool, err error) {
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(patterns) == 0 && len(strs) == 0 {
|
if len(pattern.parts) == 0 && len(strs) == 0 {
|
||||||
return true, nil
|
return true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// an empty pattern never matches a non-empty path
|
// an empty pattern never matches a non-empty path
|
||||||
if len(patterns) == 0 {
|
if len(pattern.parts) == 0 {
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(patterns) <= len(strs) {
|
if len(pattern.parts) <= len(strs) {
|
||||||
minOffset := 0
|
minOffset := 0
|
||||||
maxOffset := len(strs) - len(patterns)
|
maxOffset := len(strs) - len(pattern.parts)
|
||||||
// special case absolute patterns
|
// special case absolute patterns
|
||||||
if patterns[0].pattern == "/" {
|
if pattern.parts[0].pattern == "/" {
|
||||||
maxOffset = 0
|
maxOffset = 0
|
||||||
} else if strs[0] == "/" {
|
} else if strs[0] == "/" {
|
||||||
// skip absolute path marker if pattern is not rooted
|
// skip absolute path marker if pattern is not rooted
|
||||||
|
@ -184,12 +193,12 @@ func match(patterns Pattern, strs []string) (matched bool, err error) {
|
||||||
outer:
|
outer:
|
||||||
for offset := maxOffset; offset >= minOffset; offset-- {
|
for offset := maxOffset; offset >= minOffset; offset-- {
|
||||||
|
|
||||||
for i := len(patterns) - 1; i >= 0; i-- {
|
for i := len(pattern.parts) - 1; i >= 0; i-- {
|
||||||
var ok bool
|
var ok bool
|
||||||
if patterns[i].isSimple {
|
if pattern.parts[i].isSimple {
|
||||||
ok = patterns[i].pattern == strs[offset+i]
|
ok = pattern.parts[i].pattern == strs[offset+i]
|
||||||
} else {
|
} else {
|
||||||
ok, err = filepath.Match(patterns[i].pattern, strs[offset+i])
|
ok, err = filepath.Match(pattern.parts[i].pattern, strs[offset+i])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, errors.Wrap(err, "Match")
|
return false, errors.Wrap(err, "Match")
|
||||||
}
|
}
|
||||||
|
@ -208,9 +217,9 @@ func match(patterns Pattern, strs []string) (matched bool, err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// ParsePatterns prepares a list of patterns for use with List.
|
// ParsePatterns prepares a list of patterns for use with List.
|
||||||
func ParsePatterns(patterns []string) []Pattern {
|
func ParsePatterns(pattern []string) []Pattern {
|
||||||
patpat := make([]Pattern, 0)
|
patpat := make([]Pattern, 0)
|
||||||
for _, pat := range patterns {
|
for _, pat := range pattern {
|
||||||
if pat == "" {
|
if pat == "" {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
@ -232,7 +241,9 @@ func ListWithChild(patterns []Pattern, str string) (matched bool, childMayMatch
|
||||||
return list(patterns, true, str)
|
return list(patterns, true, str)
|
||||||
}
|
}
|
||||||
|
|
||||||
// List returns true if str matches one of the patterns. Empty patterns are ignored.
|
// list returns true if str matches one of the patterns. Empty patterns are ignored.
|
||||||
|
// Patterns prefixed by "!" are negated: any matching file excluded by a previous pattern
|
||||||
|
// will become included again.
|
||||||
func list(patterns []Pattern, checkChildMatches bool, str string) (matched bool, childMayMatch bool, err error) {
|
func list(patterns []Pattern, checkChildMatches bool, str string) (matched bool, childMayMatch bool, err error) {
|
||||||
if len(patterns) == 0 {
|
if len(patterns) == 0 {
|
||||||
return false, false, nil
|
return false, false, nil
|
||||||
|
@ -242,6 +253,12 @@ func list(patterns []Pattern, checkChildMatches bool, str string) (matched bool,
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false, false, err
|
return false, false, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hasNegatedPattern := false
|
||||||
|
for _, pat := range patterns {
|
||||||
|
hasNegatedPattern = hasNegatedPattern || pat.isNegated
|
||||||
|
}
|
||||||
|
|
||||||
for _, pat := range patterns {
|
for _, pat := range patterns {
|
||||||
m, err := match(pat, strs)
|
m, err := match(pat, strs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -258,11 +275,17 @@ func list(patterns []Pattern, checkChildMatches bool, str string) (matched bool,
|
||||||
c = true
|
c = true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if pat.isNegated {
|
||||||
|
matched = matched && !m
|
||||||
|
childMayMatch = childMayMatch && !m
|
||||||
|
} else {
|
||||||
matched = matched || m
|
matched = matched || m
|
||||||
childMayMatch = childMayMatch || c
|
childMayMatch = childMayMatch || c
|
||||||
|
|
||||||
if matched && childMayMatch {
|
if matched && childMayMatch && !hasNegatedPattern {
|
||||||
return true, true, nil
|
// without negative patterns the result cannot change any more
|
||||||
|
break
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -248,6 +248,7 @@ var filterListTests = []struct {
|
||||||
}{
|
}{
|
||||||
{[]string{}, "/foo/bar/test.go", false, false},
|
{[]string{}, "/foo/bar/test.go", false, false},
|
||||||
{[]string{"*.go"}, "/foo/bar/test.go", true, true},
|
{[]string{"*.go"}, "/foo/bar/test.go", true, true},
|
||||||
|
{[]string{"*.go"}, "/foo/bar", false, true},
|
||||||
{[]string{"*.c"}, "/foo/bar/test.go", false, true},
|
{[]string{"*.c"}, "/foo/bar/test.go", false, true},
|
||||||
{[]string{"*.go", "*.c"}, "/foo/bar/test.go", true, true},
|
{[]string{"*.go", "*.c"}, "/foo/bar/test.go", true, true},
|
||||||
{[]string{"*"}, "/foo/bar/test.go", true, true},
|
{[]string{"*"}, "/foo/bar/test.go", true, true},
|
||||||
|
@ -255,8 +256,25 @@ var filterListTests = []struct {
|
||||||
{[]string{"?"}, "/foo/bar/test.go", false, true},
|
{[]string{"?"}, "/foo/bar/test.go", false, true},
|
||||||
{[]string{"?", "x"}, "/foo/bar/x", true, true},
|
{[]string{"?", "x"}, "/foo/bar/x", true, true},
|
||||||
{[]string{"/*/*/bar/test.*"}, "/foo/bar/test.go", false, false},
|
{[]string{"/*/*/bar/test.*"}, "/foo/bar/test.go", false, false},
|
||||||
|
{[]string{"/*/*/bar/test.*"}, "/foo/bar/bar", false, true},
|
||||||
{[]string{"/*/*/bar/test.*", "*.go"}, "/foo/bar/test.go", true, true},
|
{[]string{"/*/*/bar/test.*", "*.go"}, "/foo/bar/test.go", true, true},
|
||||||
{[]string{"", "*.c"}, "/foo/bar/test.go", false, true},
|
{[]string{"", "*.c"}, "/foo/bar/test.go", false, true},
|
||||||
|
{[]string{"!**", "*.go"}, "/foo/bar/test.go", true, true},
|
||||||
|
{[]string{"!**", "*.c"}, "/foo/bar/test.go", false, true},
|
||||||
|
{[]string{"/foo/*/test.*", "!*.c"}, "/foo/bar/test.c", false, false},
|
||||||
|
{[]string{"/foo/*/test.*", "!*.c"}, "/foo/bar/test.go", true, true},
|
||||||
|
{[]string{"/foo/*/*", "!test.*", "*.c"}, "/foo/bar/test.go", false, true},
|
||||||
|
{[]string{"/foo/*/*", "!test.*", "*.c"}, "/foo/bar/test.c", true, true},
|
||||||
|
{[]string{"/foo/*/*", "!test.*", "*.c"}, "/foo/bar/file.go", true, true},
|
||||||
|
{[]string{"/**/*", "!/foo", "/foo/*", "!/foo/bar"}, "/foo/other/test.go", true, true},
|
||||||
|
{[]string{"/**/*", "!/foo", "/foo/*", "!/foo/bar"}, "/foo/bar", false, false},
|
||||||
|
{[]string{"/**/*", "!/foo", "/foo/*", "!/foo/bar"}, "/foo/bar/test.go", false, false},
|
||||||
|
{[]string{"/**/*", "!/foo", "/foo/*", "!/foo/bar"}, "/foo/bar/test.go/child", false, false},
|
||||||
|
{[]string{"/**/*", "!/foo", "/foo/*", "!/foo/bar", "/foo/bar/test*"}, "/foo/bar/test.go/child", true, true},
|
||||||
|
{[]string{"/foo/bar/*"}, "/foo", false, true},
|
||||||
|
{[]string{"/foo/bar/*", "!/foo/bar/[a-m]*"}, "/foo", false, true},
|
||||||
|
{[]string{"/foo/**/test.c"}, "/foo/bar/foo/bar/test.c", true, true},
|
||||||
|
{[]string{"/foo/*/test.c"}, "/foo/bar/foo/bar/test.c", false, false},
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestList(t *testing.T) {
|
func TestList(t *testing.T) {
|
||||||
|
|
Loading…
Reference in a new issue