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