"""Filename matching with shell patterns. fnmatch(FILENAME, PATTERN) matches according to the local convention. fnmatchcase(FILENAME, PATTERN) always takes case in account. The functions operate by translating the pattern into a regular expression. They cache the compiled regular expressions for speed. The function translate(PATTERN) returns a regular expression corresponding to PATTERN. (It does not compile it.) """ import os import posixpath import re import functools __all__ = ["filter", "fnmatch", "fnmatchcase", "translate"] # Build a thread-safe incrementing counter to help create unique regexp group # names across calls. from itertools import count _nextgroupnum = count().__next__ del count def fnmatch(name, pat): """Test whether FILENAME matches PATTERN. Patterns are Unix shell style: * matches everything ? matches any single character [seq] matches any character in seq [!seq] matches any char not in seq An initial period in FILENAME is not special. Both FILENAME and PATTERN are first case-normalized if the operating system requires it. If you don't want this, use fnmatchcase(FILENAME, PATTERN). """ name = os.path.normcase(name) pat = os.path.normcase(pat) return fnmatchcase(name, pat) @functools.lru_cache(maxsize=256, typed=True) def _compile_pattern(pat): if isinstance(pat, bytes): pat_str = str(pat, 'ISO-8859-1') res_str = translate(pat_str) res = bytes(res_str, 'ISO-8859-1') else: res = translate(pat) return re.compile(res).match def filter(names, pat): """Construct a list from those elements of the iterable NAMES that match PAT.""" result = [] pat = os.path.normcase(pat) match = _compile_pattern(pat) if os.path is posixpath: # normcase on posix is NOP. Optimize it away from the loop. for name in names: if match(name): result.append(name) else: for name in names: if match(os.path.normcase(name)): result.append(name) return result def fnmatchcase(name, pat): """Test whether FILENAME matches PATTERN, including case. This is a version of fnmatch() which doesn't case-normalize its arguments. """ match = _compile_pattern(pat) return match(name) is not None def translate(pat): """Translate a shell PATTERN to a regular expression. There is no way to quote meta-characters. """ STAR = object() res = [] add = res.append i, n = 0, len(pat) while i < n: c = pat[i] i = i+1 if c == '*': # compress consecutive `*` into one if (not res) or res[-1] is not STAR: add(STAR) elif c == '?': add('.') elif c == '[': j = i if j < n and pat[j] == '!': j = j+1 if j < n and pat[j] == ']': j = j+1 while j < n and pat[j] != ']': j = j+1 if j >= n: add('\\[') else: stuff = pat[i:j] if '-' not in stuff: stuff = stuff.replace('\\', r'\\') else: chunks = [] k = i+2 if pat[i] == '!' else i+1 while True: k = pat.find('-', k, j) if k < 0: break chunks.append(pat[i:k]) i = k+1 k = k+3 chunk = pat[i:j] if chunk: chunks.append(chunk) else: chunks[-1] += '-' # Remove empty ranges -- invalid in RE. for k in range(len(chunks)-1, 0, -1): if chunks[k-1][-1] > chunks[k][0]: chunks[k-1] = chunks[k-1][:-1] + chunks[k][1:] del chunks[k] # Escape backslashes and hyphens for set difference (--). # Hyphens that create ranges shouldn't be escaped. stuff = '-'.join(s.replace('\\', r'\\').replace('-', r'\-') for s in chunks) # Escape set operations (&&, ~~ and ||). stuff = re.sub(r'([&~|])', r'\\\1', stuff) i = j+1 if not stuff: # Empty range: never match. add('(?!)') elif stuff == '!': # Negated empty range: match any character. add('.') else: if stuff[0] == '!': stuff = '^' + stuff[1:] elif stuff[0] in ('^', '['): stuff = '\\' + stuff add(f'[{stuff}]') else: add(re.escape(c)) assert i == n # Deal with STARs. inp = res res = [] add = res.append i, n = 0, len(inp) # Fixed pieces at the start? while i < n and inp[i] is not STAR: add(inp[i]) i += 1 # Now deal with STAR fixed STAR fixed ... # For an interior `STAR fixed` pairing, we want to do a minimal # .*? match followed by `fixed`, with no possibility of backtracking. # We can't spell that directly, but can trick it into working by matching # .*?fixed # in a lookahead assertion, save the matched part in a group, then # consume that group via a backreference. If the overall match fails, # the lookahead assertion won't try alternatives. So the translation is: # (?=(?P<name>.*?fixed))(?P=name) # Group names are created as needed: g0, g1, g2, ... # The numbers are obtained from _nextgroupnum() to ensure they're unique # across calls and across threads. This is because people rely on the # undocumented ability to join multiple translate() results together via # "|" to build large regexps matching "one of many" shell patterns. while i < n: assert inp[i] is STAR i += 1 if i == n: add(".*") break assert inp[i] is not STAR fixed = [] while i < n and inp[i] is not STAR: fixed.append(inp[i]) i += 1 fixed = "".join(fixed) if i == n: add(".*") add(fixed) else: groupnum = _nextgroupnum() add(f"(?=(?P<g{groupnum}>.*?{fixed}))(?P=g{groupnum})") assert i == n res = "".join(res) return fr'(?s:{res})\Z'
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
__pycache__ | Folder | 0755 |
|
|
asyncio | Folder | 0755 |
|
|
collections | Folder | 0755 |
|
|
concurrent | Folder | 0755 |
|
|
config-3.10-x86_64-linux-gnu | Folder | 0755 |
|
|
ctypes | Folder | 0755 |
|
|
curses | Folder | 0755 |
|
|
dbm | Folder | 0755 |
|
|
distutils | Folder | 0755 |
|
|
Folder | 0755 |
|
||
encodings | Folder | 0755 |
|
|
ensurepip | Folder | 0755 |
|
|
html | Folder | 0755 |
|
|
http | Folder | 0755 |
|
|
importlib | Folder | 0755 |
|
|
json | Folder | 0755 |
|
|
lib-dynload | Folder | 0755 |
|
|
lib2to3 | Folder | 0755 |
|
|
logging | Folder | 0755 |
|
|
multiprocessing | Folder | 0755 |
|
|
pydoc_data | Folder | 0755 |
|
|
site-packages | Folder | 0755 |
|
|
sqlite3 | Folder | 0755 |
|
|
unittest | Folder | 0755 |
|
|
urllib | Folder | 0755 |
|
|
venv | Folder | 0755 |
|
|
wsgiref | Folder | 0755 |
|
|
xml | Folder | 0755 |
|
|
xmlrpc | Folder | 0755 |
|
|
zoneinfo | Folder | 0755 |
|
|
LICENSE.txt | File | 13.61 KB | 0644 |
|
__future__.py | File | 5.03 KB | 0644 |
|
__phello__.foo.py | File | 64 B | 0644 |
|
_aix_support.py | File | 3.19 KB | 0644 |
|
_bootsubprocess.py | File | 2.61 KB | 0644 |
|
_collections_abc.py | File | 31.53 KB | 0644 |
|
_compat_pickle.py | File | 8.54 KB | 0644 |
|
_compression.py | File | 5.55 KB | 0644 |
|
_markupbase.py | File | 14.31 KB | 0644 |
|
_osx_support.py | File | 21.28 KB | 0644 |
|
_py_abc.py | File | 6.04 KB | 0644 |
|
_pydecimal.py | File | 223.32 KB | 0644 |
|
_pyio.py | File | 92.25 KB | 0644 |
|
_sitebuiltins.py | File | 3.05 KB | 0644 |
|
_strptime.py | File | 24.68 KB | 0644 |
|
_sysconfigdata__linux_x86_64-linux-gnu.py | File | 40.38 KB | 0644 |
|
_sysconfigdata_d_linux_x86_64-linux-gnu.py | File | 39.81 KB | 0644 |
|
_threading_local.py | File | 7.05 KB | 0644 |
|
_weakrefset.py | File | 5.78 KB | 0644 |
|
abc.py | File | 6.37 KB | 0644 |
|
aifc.py | File | 31.84 KB | 0644 |
|
antigravity.py | File | 500 B | 0644 |
|
argparse.py | File | 96.23 KB | 0644 |
|
ast.py | File | 58.5 KB | 0644 |
|
asynchat.py | File | 11.25 KB | 0644 |
|
asyncore.py | File | 19.79 KB | 0644 |
|
base64.py | File | 20.37 KB | 0755 |
|
bdb.py | File | 31.64 KB | 0644 |
|
binhex.py | File | 14.44 KB | 0644 |
|
bisect.py | File | 3.06 KB | 0644 |
|
bz2.py | File | 11.57 KB | 0644 |
|
cProfile.py | File | 6.21 KB | 0755 |
|
calendar.py | File | 24 KB | 0644 |
|
cgi.py | File | 33.31 KB | 0755 |
|
cgitb.py | File | 11.81 KB | 0644 |
|
chunk.py | File | 5.31 KB | 0644 |
|
cmd.py | File | 14.51 KB | 0644 |
|
code.py | File | 10.37 KB | 0644 |
|
codecs.py | File | 35.85 KB | 0644 |
|
codeop.py | File | 5.48 KB | 0644 |
|
colorsys.py | File | 3.92 KB | 0644 |
|
compileall.py | File | 19.78 KB | 0644 |
|
configparser.py | File | 53.33 KB | 0644 |
|
contextlib.py | File | 25.28 KB | 0644 |
|
contextvars.py | File | 129 B | 0644 |
|
copy.py | File | 8.48 KB | 0644 |
|
copyreg.py | File | 7.25 KB | 0644 |
|
crypt.py | File | 3.76 KB | 0644 |
|
csv.py | File | 15.65 KB | 0644 |
|
dataclasses.py | File | 55.07 KB | 0644 |
|
datetime.py | File | 86.02 KB | 0644 |
|
decimal.py | File | 320 B | 0644 |
|
difflib.py | File | 81.36 KB | 0644 |
|
dis.py | File | 19.55 KB | 0644 |
|
doctest.py | File | 102.68 KB | 0644 |
|
enum.py | File | 38.9 KB | 0644 |
|
filecmp.py | File | 9.94 KB | 0644 |
|
fileinput.py | File | 16.06 KB | 0644 |
|
fnmatch.py | File | 6.56 KB | 0644 |
|
fractions.py | File | 27.58 KB | 0644 |
|
ftplib.py | File | 34.66 KB | 0644 |
|
functools.py | File | 37.18 KB | 0644 |
|
genericpath.py | File | 4.86 KB | 0644 |
|
getopt.py | File | 7.31 KB | 0644 |
|
getpass.py | File | 5.85 KB | 0644 |
|
gettext.py | File | 26.63 KB | 0644 |
|
glob.py | File | 7.7 KB | 0644 |
|
graphlib.py | File | 9.35 KB | 0644 |
|
gzip.py | File | 21.34 KB | 0644 |
|
hashlib.py | File | 9.99 KB | 0644 |
|
heapq.py | File | 22.34 KB | 0644 |
|
hmac.py | File | 7.54 KB | 0644 |
|
imaplib.py | File | 53.58 KB | 0644 |
|
imghdr.py | File | 3.72 KB | 0644 |
|
imp.py | File | 10.34 KB | 0644 |
|
inspect.py | File | 121.46 KB | 0644 |
|
io.py | File | 4.1 KB | 0644 |
|
ipaddress.py | File | 75.96 KB | 0644 |
|
keyword.py | File | 1.04 KB | 0644 |
|
linecache.py | File | 5.56 KB | 0644 |
|
locale.py | File | 76.29 KB | 0644 |
|
lzma.py | File | 12.97 KB | 0644 |
|
mailbox.py | File | 76.95 KB | 0644 |
|
mailcap.py | File | 8.9 KB | 0644 |
|
mimetypes.py | File | 22.01 KB | 0644 |
|
modulefinder.py | File | 23.83 KB | 0644 |
|
netrc.py | File | 5.61 KB | 0644 |
|
nntplib.py | File | 40.06 KB | 0644 |
|
ntpath.py | File | 28.79 KB | 0644 |
|
nturl2path.py | File | 2.82 KB | 0644 |
|
numbers.py | File | 10.11 KB | 0644 |
|
opcode.py | File | 5.76 KB | 0644 |
|
operator.py | File | 10.5 KB | 0644 |
|
optparse.py | File | 58.95 KB | 0644 |
|
os.py | File | 38.63 KB | 0644 |
|
pathlib.py | File | 48.41 KB | 0644 |
|
pdb.py | File | 61.76 KB | 0755 |
|
pickle.py | File | 63.43 KB | 0644 |
|
pickletools.py | File | 91.29 KB | 0644 |
|
pipes.py | File | 8.71 KB | 0644 |
|
pkgutil.py | File | 24 KB | 0644 |
|
platform.py | File | 41.05 KB | 0755 |
|
plistlib.py | File | 27.69 KB | 0644 |
|
poplib.py | File | 14.84 KB | 0644 |
|
posixpath.py | File | 15.87 KB | 0644 |
|
pprint.py | File | 23.87 KB | 0644 |
|
profile.py | File | 22.36 KB | 0755 |
|
pstats.py | File | 28.64 KB | 0644 |
|
pty.py | File | 5.09 KB | 0644 |
|
py_compile.py | File | 7.71 KB | 0644 |
|
pyclbr.py | File | 11.13 KB | 0644 |
|
pydoc.py | File | 107.03 KB | 0755 |
|
queue.py | File | 11.23 KB | 0644 |
|
quopri.py | File | 7.11 KB | 0755 |
|
random.py | File | 32.44 KB | 0644 |
|
re.py | File | 15.49 KB | 0644 |
|
reprlib.py | File | 5.14 KB | 0644 |
|
rlcompleter.py | File | 7.63 KB | 0644 |
|
runpy.py | File | 12.8 KB | 0644 |
|
sched.py | File | 6.2 KB | 0644 |
|
secrets.py | File | 1.99 KB | 0644 |
|
selectors.py | File | 19.08 KB | 0644 |
|
shelve.py | File | 8.36 KB | 0644 |
|
shlex.py | File | 13.18 KB | 0644 |
|
shutil.py | File | 53.29 KB | 0644 |
|
signal.py | File | 2.38 KB | 0644 |
|
site.py | File | 22.39 KB | 0644 |
|
smtpd.py | File | 34.35 KB | 0755 |
|
smtplib.py | File | 44.37 KB | 0755 |
|
sndhdr.py | File | 6.93 KB | 0644 |
|
socket.py | File | 36.14 KB | 0644 |
|
socketserver.py | File | 26.66 KB | 0644 |
|
sre_compile.py | File | 27.32 KB | 0644 |
|
sre_constants.py | File | 7.01 KB | 0644 |
|
sre_parse.py | File | 39.82 KB | 0644 |
|
ssl.py | File | 52.63 KB | 0644 |
|
stat.py | File | 5.36 KB | 0644 |
|
statistics.py | File | 42.19 KB | 0644 |
|
string.py | File | 10.32 KB | 0644 |
|
stringprep.py | File | 12.61 KB | 0644 |
|
struct.py | File | 257 B | 0644 |
|
subprocess.py | File | 82.93 KB | 0644 |
|
sunau.py | File | 17.73 KB | 0644 |
|
symtable.py | File | 9.98 KB | 0644 |
|
sysconfig.py | File | 26.96 KB | 0644 |
|
tabnanny.py | File | 11.05 KB | 0755 |
|
tarfile.py | File | 105.13 KB | 0755 |
|
telnetlib.py | File | 22.71 KB | 0644 |
|
tempfile.py | File | 28.78 KB | 0644 |
|
textwrap.py | File | 19.31 KB | 0644 |
|
this.py | File | 1003 B | 0644 |
|
threading.py | File | 55.41 KB | 0644 |
|
timeit.py | File | 13.19 KB | 0755 |
|
token.py | File | 2.33 KB | 0644 |
|
tokenize.py | File | 25.31 KB | 0644 |
|
trace.py | File | 28.54 KB | 0755 |
|
traceback.py | File | 25.61 KB | 0644 |
|
tracemalloc.py | File | 17.62 KB | 0644 |
|
tty.py | File | 879 B | 0644 |
|
types.py | File | 9.88 KB | 0644 |
|
typing.py | File | 90.39 KB | 0644 |
|
uu.py | File | 7.11 KB | 0644 |
|
uuid.py | File | 26.86 KB | 0644 |
|
warnings.py | File | 19.23 KB | 0644 |
|
wave.py | File | 17.58 KB | 0644 |
|
weakref.py | File | 21.05 KB | 0644 |
|
webbrowser.py | File | 23.69 KB | 0755 |
|
xdrlib.py | File | 5.77 KB | 0644 |
|
zipapp.py | File | 7.36 KB | 0644 |
|
zipfile.py | File | 88.09 KB | 0644 |
|
zipimport.py | File | 30.17 KB | 0644 |
|