glob.py 1.96 KB
Newer Older
Guido van Rossum's avatar
Guido van Rossum committed
1
"""Filename globbing utility."""
Guido van Rossum's avatar
Guido van Rossum committed
2

Guido van Rossum's avatar
Guido van Rossum committed
3
import os
4
import fnmatch
5
import re
Guido van Rossum's avatar
Guido van Rossum committed
6

Johannes Gijsbers's avatar
Johannes Gijsbers committed
7
__all__ = ["glob", "iglob"]
Guido van Rossum's avatar
Guido van Rossum committed
8

Guido van Rossum's avatar
Guido van Rossum committed
9
def glob(pathname):
Tim Peters's avatar
Tim Peters committed
10 11 12 13
    """Return a list of paths matching a pathname pattern.

    The pattern may contain simple shell-style wildcards a la fnmatch.

Johannes Gijsbers's avatar
Johannes Gijsbers committed
14 15 16 17 18 19 20 21
    """
    return list(iglob(pathname))

def iglob(pathname):
    """Return a list of paths matching a pathname pattern.

    The pattern may contain simple shell-style wildcards a la fnmatch.

Tim Peters's avatar
Tim Peters committed
22 23
    """
    if not has_magic(pathname):
24
        if os.path.lexists(pathname):
Johannes Gijsbers's avatar
Johannes Gijsbers committed
25 26
            yield pathname
        return
Tim Peters's avatar
Tim Peters committed
27
    dirname, basename = os.path.split(pathname)
28
    if not dirname:
Johannes Gijsbers's avatar
Johannes Gijsbers committed
29 30 31 32 33
        for name in glob1(os.curdir, basename):
            yield name
        return
    if has_magic(dirname):
        dirs = iglob(dirname)
Tim Peters's avatar
Tim Peters committed
34
    else:
Johannes Gijsbers's avatar
Johannes Gijsbers committed
35 36 37
        dirs = [dirname]
    if has_magic(basename):
        glob_in_dir = glob1
Tim Peters's avatar
Tim Peters committed
38
    else:
Johannes Gijsbers's avatar
Johannes Gijsbers committed
39 40 41 42 43 44 45 46
        glob_in_dir = glob0
    for dirname in dirs:
        for name in glob_in_dir(dirname, basename):
            yield os.path.join(dirname, name)

# These 2 helper functions non-recursively glob inside a literal directory.
# They return a list of basenames. `glob1` accepts a pattern while `glob0`
# takes a literal basename (so it only has to check for its existence).
Guido van Rossum's avatar
Guido van Rossum committed
47 48

def glob1(dirname, pattern):
Johannes Gijsbers's avatar
Johannes Gijsbers committed
49 50
    if not dirname:
        dirname = os.curdir
Tim Peters's avatar
Tim Peters committed
51 52 53 54
    try:
        names = os.listdir(dirname)
    except os.error:
        return []
55 56 57
    if pattern[0]!='.':
        names=filter(lambda x: x[0]!='.',names)
    return fnmatch.filter(names,pattern)
Guido van Rossum's avatar
Guido van Rossum committed
58

Johannes Gijsbers's avatar
Johannes Gijsbers committed
59 60 61 62
def glob0(dirname, basename):
    if basename == '':
        # `os.path.split()` returns an empty basename for paths ending with a
        # directory separator.  'q*x/' should match only directories.
63
        if os.path.isdir(dirname):
Johannes Gijsbers's avatar
Johannes Gijsbers committed
64 65 66 67 68 69
            return [basename]
    else:
        if os.path.lexists(os.path.join(dirname, basename)):
            return [basename]
    return []

70

71
magic_check = re.compile('[*?[]')
72

Guido van Rossum's avatar
Guido van Rossum committed
73
def has_magic(s):
Tim Peters's avatar
Tim Peters committed
74
    return magic_check.search(s) is not None