2014-09-04 13:29:53 -07:00
|
|
|
// Copyright (C) 2014 Jakob Borg and Contributors (see the CONTRIBUTORS file).
|
2014-09-29 12:43:32 -07:00
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify it
|
|
|
|
// under the terms of the GNU General Public License as published by the Free
|
|
|
|
// Software Foundation, either version 3 of the License, or (at your option)
|
|
|
|
// any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
// more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License along
|
|
|
|
// with this program. If not, see <http://www.gnu.org/licenses/>.
|
2014-09-04 13:29:53 -07:00
|
|
|
|
|
|
|
package ignore
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
|
2014-09-22 12:42:11 -07:00
|
|
|
"github.com/syncthing/syncthing/internal/fnmatch"
|
2014-09-04 13:29:53 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
type Pattern struct {
|
|
|
|
match *regexp.Regexp
|
|
|
|
include bool
|
|
|
|
}
|
|
|
|
|
|
|
|
type Patterns []Pattern
|
|
|
|
|
|
|
|
func Load(file string) (Patterns, error) {
|
|
|
|
seen := make(map[string]bool)
|
|
|
|
return loadIgnoreFile(file, seen)
|
|
|
|
}
|
|
|
|
|
|
|
|
func Parse(r io.Reader, file string) (Patterns, error) {
|
|
|
|
seen := map[string]bool{
|
|
|
|
file: true,
|
|
|
|
}
|
|
|
|
return parseIgnoreFile(r, file, seen)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l Patterns) Match(file string) bool {
|
|
|
|
for _, pattern := range l {
|
|
|
|
if pattern.match.MatchString(file) {
|
|
|
|
return pattern.include
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func loadIgnoreFile(file string, seen map[string]bool) (Patterns, error) {
|
|
|
|
if seen[file] {
|
|
|
|
return nil, fmt.Errorf("Multiple include of ignore file %q", file)
|
|
|
|
}
|
|
|
|
seen[file] = true
|
|
|
|
|
|
|
|
fd, err := os.Open(file)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer fd.Close()
|
|
|
|
|
|
|
|
return parseIgnoreFile(fd, file, seen)
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseIgnoreFile(fd io.Reader, currentFile string, seen map[string]bool) (Patterns, error) {
|
|
|
|
var exps Patterns
|
|
|
|
|
|
|
|
addPattern := func(line string) error {
|
|
|
|
include := true
|
|
|
|
if strings.HasPrefix(line, "!") {
|
|
|
|
line = line[1:]
|
|
|
|
include = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if strings.HasPrefix(line, "/") {
|
|
|
|
// Pattern is rooted in the current dir only
|
|
|
|
exp, err := fnmatch.Convert(line[1:], fnmatch.FNM_PATHNAME)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Invalid pattern %q in ignore file", line)
|
|
|
|
}
|
|
|
|
exps = append(exps, Pattern{exp, include})
|
|
|
|
} else if strings.HasPrefix(line, "**/") {
|
|
|
|
// Add the pattern as is, and without **/ so it matches in current dir
|
|
|
|
exp, err := fnmatch.Convert(line, fnmatch.FNM_PATHNAME)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Invalid pattern %q in ignore file", line)
|
|
|
|
}
|
|
|
|
exps = append(exps, Pattern{exp, include})
|
|
|
|
|
|
|
|
exp, err = fnmatch.Convert(line[3:], fnmatch.FNM_PATHNAME)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Invalid pattern %q in ignore file", line)
|
|
|
|
}
|
|
|
|
exps = append(exps, Pattern{exp, include})
|
|
|
|
} else if strings.HasPrefix(line, "#include ") {
|
|
|
|
includeFile := filepath.Join(filepath.Dir(currentFile), line[len("#include "):])
|
|
|
|
includes, err := loadIgnoreFile(includeFile, seen)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
} else {
|
|
|
|
exps = append(exps, includes...)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Path name or pattern, add it so it matches files both in
|
|
|
|
// current directory and subdirs.
|
|
|
|
exp, err := fnmatch.Convert(line, fnmatch.FNM_PATHNAME)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Invalid pattern %q in ignore file", line)
|
|
|
|
}
|
|
|
|
exps = append(exps, Pattern{exp, include})
|
|
|
|
|
|
|
|
exp, err = fnmatch.Convert("**/"+line, fnmatch.FNM_PATHNAME)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Invalid pattern %q in ignore file", line)
|
|
|
|
}
|
|
|
|
exps = append(exps, Pattern{exp, include})
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
scanner := bufio.NewScanner(fd)
|
|
|
|
var err error
|
|
|
|
for scanner.Scan() {
|
|
|
|
line := strings.TrimSpace(scanner.Text())
|
|
|
|
switch {
|
|
|
|
case line == "":
|
|
|
|
continue
|
2014-09-16 14:22:21 -07:00
|
|
|
case strings.HasPrefix(line, "//"):
|
|
|
|
continue
|
2014-09-04 13:29:53 -07:00
|
|
|
case strings.HasPrefix(line, "#"):
|
|
|
|
err = addPattern(line)
|
|
|
|
case strings.HasSuffix(line, "/**"):
|
|
|
|
err = addPattern(line)
|
|
|
|
case strings.HasSuffix(line, "/"):
|
|
|
|
err = addPattern(line)
|
|
|
|
if err == nil {
|
|
|
|
err = addPattern(line + "**")
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
err = addPattern(line)
|
|
|
|
if err == nil {
|
|
|
|
err = addPattern(line + "/**")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return exps, nil
|
|
|
|
}
|