// Copyright 2020 Google LLC // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // https://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // Package glob provides file globbing utilities package glob import ( "bytes" "encoding/json" "fmt" "io/ioutil" "os" "path/filepath" "strings" "dawn.googlesource.com/tint/tools/src/match" ) // Scan walks all files and subdirectories from root, returning those // that Config.shouldExamine() returns true for. func Scan(root string, cfg Config) ([]string, error) { files := []string{} err := filepath.Walk(root, func(path string, info os.FileInfo, err error) error { rel, err := filepath.Rel(root, path) if err != nil { rel = path } if rel == ".git" { return filepath.SkipDir } if !cfg.shouldExamine(root, path) { return nil } if !info.IsDir() { files = append(files, rel) } return nil }) if err != nil { return nil, err } return files, nil } // Configs is a slice of Config. type Configs []Config // Config is used to parse the JSON configuration file. type Config struct { // Paths holds a number of JSON objects that contain either a "includes" or // "excludes" key to an array of path patterns. // Each path pattern is considered in turn to either include or exclude the // file path for license scanning. Pattern use forward-slashes '/' for // directory separators, and may use the following wildcards: // ? - matches any single non-separator character // * - matches any sequence of non-separator characters // ** - matches any sequence of characters including separators // // Rules are processed in the order in which they are declared, with later // rules taking precedence over earlier rules. // // All files are excluded before the first rule is evaluated. // // Example: // // { // "paths": [ // { "exclude": [ "out/*", "build/*" ] }, // { "include": [ "out/foo.txt" ] } // ], // } Paths searchRules } // LoadConfig loads a config file at path. func LoadConfig(path string) (Config, error) { cfgBody, err := ioutil.ReadFile(path) if err != nil { return Config{}, err } return ParseConfig(string(cfgBody)) } // ParseConfig parses the config from a JSON string. func ParseConfig(config string) (Config, error) { d := json.NewDecoder(strings.NewReader(config)) cfg := Config{} if err := d.Decode(&cfg); err != nil { return Config{}, err } return cfg, nil } // MustParseConfig parses the config from a JSON string, panicing if the config // does not parse func MustParseConfig(config string) Config { d := json.NewDecoder(strings.NewReader(config)) cfg := Config{} if err := d.Decode(&cfg); err != nil { panic(fmt.Errorf("Failed to parse config: %w\nConfig:\n%v", err, config)) } return cfg } // rule is a search path predicate. // root is the project relative path. // cond is the value to return if the rule doesn't either include or exclude. type rule func(path string, cond bool) bool // searchRules is a ordered list of search rules. // searchRules is its own type as it has to perform custom JSON unmarshalling. type searchRules []rule // UnmarshalJSON unmarshals the array of rules in the form: // { "include": [ ... ] } or { "exclude": [ ... ] } func (l *searchRules) UnmarshalJSON(body []byte) error { type parsed struct { Include []string Exclude []string } p := []parsed{} if err := json.NewDecoder(bytes.NewReader(body)).Decode(&p); err != nil { return err } *l = searchRules{} for _, rule := range p { rule := rule switch { case len(rule.Include) > 0 && len(rule.Exclude) > 0: return fmt.Errorf("Rule cannot contain both include and exclude") case len(rule.Include) > 0: tests := make([]match.Test, len(rule.Include)) for i, pattern := range rule.Include { test, err := match.New(pattern) if err != nil { return err } tests[i] = test } *l = append(*l, func(path string, cond bool) bool { for _, test := range tests { if test(path) { return true } } return cond }) case len(rule.Exclude) > 0: tests := make([]match.Test, len(rule.Exclude)) for i, pattern := range rule.Exclude { test, err := match.New(pattern) if err != nil { return err } tests[i] = test } *l = append(*l, func(path string, cond bool) bool { for _, test := range tests { if test(path) { return false } } return cond }) } } return nil } // shouldExamine returns true if the file at absPath should be scanned. func (c Config) shouldExamine(root, absPath string) bool { root = filepath.ToSlash(root) // Canonicalize absPath = filepath.ToSlash(absPath) // Canonicalize relPath, err := filepath.Rel(root, absPath) if err != nil { return false } relPath = filepath.ToSlash(relPath) // Canonicalize res := false for _, rule := range c.Paths { res = rule(relPath, res) } return res }