| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869 |
- # 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)
|