blob: 7bef6f190dac54a762dfc741e85ed58aa320abd7 [file] [log] [blame]
# Copyright (c) 2021 Project CHIP Authors
#
# 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
#
# http://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.
def _GlobMatch(glob: str, value: str) -> bool:
"""Does limited glob string matching with the following support:
"*" matches 0 or more characters
"?" matches any 1 character
"{x,y,z}" matches x or y or z
There are limitations in the current implementation:
- no support for escapes (cannot match literal *, ? or {)
- no support for nested {} (like "{a,foo{b,c}bar,d}" used
to match "{a,foobbar,foocbar,d}")
"""
glob = glob.replace('**', '*')
while glob and value:
if glob[0] == '?':
glob, value = glob[1:], value[1:]
elif glob[0] == '*':
for idx in range(len(value)+1):
if _GlobMatch(glob[1:], value[idx:]):
return True
return False
elif glob[0] == '{':
# Format is comma separated values between {}:
# NOTE: this does NOT support nested {} at the moment
closing_idx = glob.find('}')
if not closing_idx:
raise Exception("Malformed glob expression: missing '}'")
for choice in glob[1: closing_idx].split(','):
if _GlobMatch(choice + glob[closing_idx+1:], value):
return True
return False
else:
if glob[0] != value[0]:
return False
glob, value = glob[1:], value[1:]
# if value is empty it has a chance to match subgroups
if not value and glob.startswith('{') and glob.endswith('}'):
for choice in glob[1: -1].split(','):
if _GlobMatch(choice, value):
return True
return glob == '*' or (not glob and not value)
class GlobMatcher:
def __init__(self, glob: str):
self.glob = glob
def matches(self, value: str) -> bool:
return _GlobMatch(self.glob, value)