xref: /aosp_15_r20/build/soong/cmd/diff_target_files/glob.go (revision 333d2b3687b3a337dbcca9d65000bca186795e39)
1*333d2b36SAndroid Build Coastguard Worker// Copyright 2019 Google Inc. All rights reserved.
2*333d2b36SAndroid Build Coastguard Worker//
3*333d2b36SAndroid Build Coastguard Worker// Licensed under the Apache License, Version 2.0 (the "License");
4*333d2b36SAndroid Build Coastguard Worker// you may not use this file except in compliance with the License.
5*333d2b36SAndroid Build Coastguard Worker// You may obtain a copy of the License at
6*333d2b36SAndroid Build Coastguard Worker//
7*333d2b36SAndroid Build Coastguard Worker//     http://www.apache.org/licenses/LICENSE-2.0
8*333d2b36SAndroid Build Coastguard Worker//
9*333d2b36SAndroid Build Coastguard Worker// Unless required by applicable law or agreed to in writing, software
10*333d2b36SAndroid Build Coastguard Worker// distributed under the License is distributed on an "AS IS" BASIS,
11*333d2b36SAndroid Build Coastguard Worker// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*333d2b36SAndroid Build Coastguard Worker// See the License for the specific language governing permissions and
13*333d2b36SAndroid Build Coastguard Worker// limitations under the License.
14*333d2b36SAndroid Build Coastguard Worker
15*333d2b36SAndroid Build Coastguard Workerpackage main
16*333d2b36SAndroid Build Coastguard Worker
17*333d2b36SAndroid Build Coastguard Workerimport (
18*333d2b36SAndroid Build Coastguard Worker	"errors"
19*333d2b36SAndroid Build Coastguard Worker	"path/filepath"
20*333d2b36SAndroid Build Coastguard Worker	"strings"
21*333d2b36SAndroid Build Coastguard Worker)
22*333d2b36SAndroid Build Coastguard Worker
23*333d2b36SAndroid Build Coastguard Worker// Match returns true if name matches pattern using the same rules as filepath.Match, but supporting
24*333d2b36SAndroid Build Coastguard Worker// recursive globs (**).
25*333d2b36SAndroid Build Coastguard Workerfunc Match(pattern, name string) (bool, error) {
26*333d2b36SAndroid Build Coastguard Worker	if filepath.Base(pattern) == "**" {
27*333d2b36SAndroid Build Coastguard Worker		return false, errors.New("pattern has '**' as last path element")
28*333d2b36SAndroid Build Coastguard Worker	}
29*333d2b36SAndroid Build Coastguard Worker
30*333d2b36SAndroid Build Coastguard Worker	patternDir := pattern[len(pattern)-1] == '/'
31*333d2b36SAndroid Build Coastguard Worker	nameDir := name[len(name)-1] == '/'
32*333d2b36SAndroid Build Coastguard Worker
33*333d2b36SAndroid Build Coastguard Worker	if patternDir != nameDir {
34*333d2b36SAndroid Build Coastguard Worker		return false, nil
35*333d2b36SAndroid Build Coastguard Worker	}
36*333d2b36SAndroid Build Coastguard Worker
37*333d2b36SAndroid Build Coastguard Worker	if nameDir {
38*333d2b36SAndroid Build Coastguard Worker		name = name[:len(name)-1]
39*333d2b36SAndroid Build Coastguard Worker		pattern = pattern[:len(pattern)-1]
40*333d2b36SAndroid Build Coastguard Worker	}
41*333d2b36SAndroid Build Coastguard Worker
42*333d2b36SAndroid Build Coastguard Worker	for {
43*333d2b36SAndroid Build Coastguard Worker		var patternFile, nameFile string
44*333d2b36SAndroid Build Coastguard Worker		pattern, patternFile = filepath.Dir(pattern), filepath.Base(pattern)
45*333d2b36SAndroid Build Coastguard Worker
46*333d2b36SAndroid Build Coastguard Worker		if patternFile == "**" {
47*333d2b36SAndroid Build Coastguard Worker			if strings.Contains(pattern, "**") {
48*333d2b36SAndroid Build Coastguard Worker				return false, errors.New("pattern contains multiple '**'")
49*333d2b36SAndroid Build Coastguard Worker			}
50*333d2b36SAndroid Build Coastguard Worker			// Test if the any prefix of name matches the part of the pattern before **
51*333d2b36SAndroid Build Coastguard Worker			for {
52*333d2b36SAndroid Build Coastguard Worker				if name == "." || name == "/" {
53*333d2b36SAndroid Build Coastguard Worker					return name == pattern, nil
54*333d2b36SAndroid Build Coastguard Worker				}
55*333d2b36SAndroid Build Coastguard Worker				if match, err := filepath.Match(pattern, name); err != nil {
56*333d2b36SAndroid Build Coastguard Worker					return false, err
57*333d2b36SAndroid Build Coastguard Worker				} else if match {
58*333d2b36SAndroid Build Coastguard Worker					return true, nil
59*333d2b36SAndroid Build Coastguard Worker				}
60*333d2b36SAndroid Build Coastguard Worker				name = filepath.Dir(name)
61*333d2b36SAndroid Build Coastguard Worker			}
62*333d2b36SAndroid Build Coastguard Worker		} else if strings.Contains(patternFile, "**") {
63*333d2b36SAndroid Build Coastguard Worker			return false, errors.New("pattern contains other characters between '**' and path separator")
64*333d2b36SAndroid Build Coastguard Worker		}
65*333d2b36SAndroid Build Coastguard Worker
66*333d2b36SAndroid Build Coastguard Worker		name, nameFile = filepath.Dir(name), filepath.Base(name)
67*333d2b36SAndroid Build Coastguard Worker
68*333d2b36SAndroid Build Coastguard Worker		if nameFile == "." && patternFile == "." {
69*333d2b36SAndroid Build Coastguard Worker			return true, nil
70*333d2b36SAndroid Build Coastguard Worker		} else if nameFile == "/" && patternFile == "/" {
71*333d2b36SAndroid Build Coastguard Worker			return true, nil
72*333d2b36SAndroid Build Coastguard Worker		} else if nameFile == "." || patternFile == "." || nameFile == "/" || patternFile == "/" {
73*333d2b36SAndroid Build Coastguard Worker			return false, nil
74*333d2b36SAndroid Build Coastguard Worker		}
75*333d2b36SAndroid Build Coastguard Worker
76*333d2b36SAndroid Build Coastguard Worker		match, err := filepath.Match(patternFile, nameFile)
77*333d2b36SAndroid Build Coastguard Worker		if err != nil || !match {
78*333d2b36SAndroid Build Coastguard Worker			return match, err
79*333d2b36SAndroid Build Coastguard Worker		}
80*333d2b36SAndroid Build Coastguard Worker	}
81*333d2b36SAndroid Build Coastguard Worker}
82