filelist.py 12.9 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
"""distutils.filelist

Provides the FileList class, used for poking about the filesystem
and building lists of files.
"""

# created 2000/07/17, Rene Liebscher (as template.py)
# most parts taken from commands/sdist.py
# renamed 2000/07/29 (to filelist.py) and officially added to
#  the Distutils source, Greg Ward 

__revision__ = "$Id$"

import sys, os, string, re
import fnmatch
from types import *
from glob import glob
from distutils.util import convert_path
19
from distutils.errors import DistutilsTemplateError, DistutilsInternalError
20 21 22

class FileList:

23 24 25 26 27 28 29 30 31 32 33 34 35
    """A list of files built by on exploring the filesystem and filtered by
    applying various patterns to what we find there.

    Instance attributes:
      dir
        directory from which files will be taken -- only used if
        'allfiles' not supplied to constructor
      files
        list of filenames currently being built/filtered/manipulated
      allfiles
        complete list of files under consideration (ie. without any
        filtering applied)
    """
36 37 38 39 40 41 42 43 44 45 46 47 48 49

    def __init__(self, 
                 files=[], 
                 dir=os.curdir, 
                 allfiles=None, 
                 warn=None, 
                 debug_print=None):
        # use standard warning and debug functions, if no other given
        if warn is None: warn = self.__warn 
        if debug_print is None: debug_print = self.__debug_print
        self.warn = warn
        self.debug_print = debug_print
        self.files = files
        self.dir = dir
50 51

        # if None, 'allfiles' will be filled when used for first time
52 53 54 55 56
        self.allfiles = allfiles 


    # standard warning and debug functions, if no other given
    def __warn (self, msg):
57
        sys.stderr.write ("warning: %s\n" % msg)
58 59 60 61 62 63 64 65 66 67
        
    def __debug_print (self, msg):
        """Print 'msg' to stdout if the global DEBUG (taken from the
        DISTUTILS_DEBUG environment variable) flag is true.
        """
        from distutils.core import DEBUG
        if DEBUG:
            print msg

    
68
    def _parse_template_line (self, line):
69 70 71
        words = string.split (line)
        action = words[0]

72 73 74 75
        patterns = dir = dir_pattern = None

        if action in ('include', 'exclude',
                      'global-include', 'global-exclude'):
76
            if len (words) < 2:
77 78
                raise DistutilsTemplateError, \
                      "'%s' expects <pattern1> <pattern2> ..." % action
79

80
            patterns = map(convert_path, words[1:])
81

82
        elif action in ('recursive-include', 'recursive-exclude'):
83
            if len (words) < 3:
84 85
                raise DistutilsTemplateError, \
                      "'%s' expects <dir> <pattern1> <pattern2> ..." % action
86

87
            dir = convert_path(words[1])
88
            patterns = map(convert_path, words[2:])
89

90
        elif action in ('graft', 'prune'):
91
            if len (words) != 2:
92 93
                raise DistutilsTemplateError, \
                     "'%s' expects a single <dir_pattern>" % action
94

95
            dir_pattern = convert_path(words[1])
96 97

        else:
98 99 100 101 102 103 104 105 106 107 108 109 110 111 112
            raise DistutilsTemplateError, "unknown action '%s'" % action

        return (action, pattern, dir, dir_pattern)

    # _parse_template_line ()
    

    def process_template_line (self, line):    

        # Parse the line: split it up, make sure the right number of words
        # are there, and return the relevant words.  'action' is always
        # defined: it's the first word of the line.  Which of the other
        # three are defined depends on the action; it'll be either
        # patterns, (dir and patterns), or (dir_pattern).
        (action, patterns, dir, dir_pattern) = self._parse_template_line(line)
113 114 115

        # OK, now we know that the action is valid and we have the
        # right number of words on the line for that action -- so we
116
        # can proceed with minimal error-checking.
117
        if action == 'include':
118 119
            self.debug_print("include " + string.join(patterns))
            for pattern in patterns:
120
                if not self.select_pattern (pattern, anchor=1):
121
                    self.warn("no files found matching '%s'" % pattern)
122 123

        elif action == 'exclude':
124 125
            self.debug_print("exclude " + string.join(patterns))
            for pattern in patterns:
126
                if not self.exclude_pattern (pattern, anchor=1):
127
                    self.warn(
128 129 130 131
                        "no previously-included files found matching '%s'"%
                        pattern)

        elif action == 'global-include':
132 133
            self.debug_print("global-include " + string.join(patterns))
            for pattern in patterns:
134 135
                if not self.select_pattern (pattern, anchor=0):
                    self.warn (("no files found matching '%s' " +
136 137
                                "anywhere in distribution") %
                               pattern)
138 139

        elif action == 'global-exclude':
140 141
            self.debug_print("global-exclude " + string.join(patterns))
            for pattern in patterns:
142
                if not self.exclude_pattern (pattern, anchor=0):
143 144 145
                    self.warn(("no previously-included files matching '%s' " +
                               "found anywhere in distribution") %
                              pattern)
146 147 148

        elif action == 'recursive-include':
            self.debug_print("recursive-include %s %s" %
149 150
                             (dir, string.join(patterns)))
            for pattern in patterns:
151 152
                if not self.select_pattern (pattern, prefix=dir):
                    self.warn (("no files found matching '%s' " +
153 154
                                "under directory '%s'") %
                               (pattern, dir))
155 156 157

        elif action == 'recursive-exclude':
            self.debug_print("recursive-exclude %s %s" %
158 159
                             (dir, string.join(patterns)))
            for pattern in patterns:
160
                if not self.exclude_pattern(pattern, prefix=dir):
161 162 163
                    self.warn(("no previously-included files matching '%s' " +
                               "found under directory '%s'") %
                              (pattern, dir))
164 165 166 167

        elif action == 'graft':
            self.debug_print("graft " + dir_pattern)
            if not self.select_pattern(None, prefix=dir_pattern):
168
                self.warn ("no directories found matching '%s'" % dir_pattern)
169 170 171 172

        elif action == 'prune':
            self.debug_print("prune " + dir_pattern)
            if not self.exclude_pattern(None, prefix=dir_pattern):
173 174 175
                self.warn(("no previously-included directories found " +
                           "matching '%s'") %
                          dir_pattern)
176
        else:
177
            raise DistutilsInternalError, \
178
                  "this cannot happen: invalid action '%s'" % action
179

180
    # process_template_line ()
181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359


    def select_pattern (self, pattern,
                        anchor=1, prefix=None, is_regex=0):
        """Select strings (presumably filenames) from 'files' that match
        'pattern', a Unix-style wildcard (glob) pattern.  Patterns are not
        quite the same as implemented by the 'fnmatch' module: '*' and '?'
        match non-special characters, where "special" is platform-dependent:
        slash on Unix, colon, slash, and backslash on DOS/Windows, and colon on
        Mac OS.

        If 'anchor' is true (the default), then the pattern match is more
        stringent: "*.py" will match "foo.py" but not "foo/bar.py".  If
        'anchor' is false, both of these will match.

        If 'prefix' is supplied, then only filenames starting with 'prefix'
        (itself a pattern) and ending with 'pattern', with anything in between
        them, will match.  'anchor' is ignored in this case.

        If 'is_regex' is true, 'anchor' and 'prefix' are ignored, and
        'pattern' is assumed to be either a string containing a regex or a
        regex object -- no translation is done, the regex is just compiled
        and used as-is.

        Selected strings will be added to self.files.

        Return 1 if files are found.
        """
        files_found = 0
        pattern_re = translate_pattern (pattern, anchor, prefix, is_regex)
        self.debug_print("select_pattern: applying regex r'%s'" %
                         pattern_re.pattern)

        # delayed loading of allfiles list
        if self.allfiles is None: self.allfiles = findall (self.dir)

        for name in self.allfiles:
            if pattern_re.search (name):
                self.debug_print(" adding " + name)
                self.files.append (name)
                files_found = 1
    
        return files_found

    # select_pattern ()


    def exclude_pattern (self, pattern,
                         anchor=1, prefix=None, is_regex=0):
        """Remove strings (presumably filenames) from 'files' that match
        'pattern'.  Other parameters are the same as for
        'select_pattern()', above.  
        The list 'self.files' is modified in place.
        Return 1 if files are found.
        """
        files_found = 0
        pattern_re = translate_pattern (pattern, anchor, prefix, is_regex)
        self.debug_print("exclude_pattern: applying regex r'%s'" %
                         pattern_re.pattern)
        for i in range (len(self.files)-1, -1, -1):
            if pattern_re.search (self.files[i]):
                self.debug_print(" removing " + self.files[i])
                del self.files[i]
                files_found = 1
    
        return files_found

    # exclude_pattern ()


    def recursive_exclude_pattern (self, dir, pattern=None):
        """Remove filenames from 'self.files' that are under 'dir' and
        whose basenames match 'pattern'.
        Return 1 if files are found.
        """
        files_found = 0
        self.debug_print("recursive_exclude_pattern: dir=%s, pattern=%s" %
                         (dir, pattern))
        if pattern is None:
            pattern_re = None
        else:
            pattern_re = translate_pattern (pattern)

        for i in range (len (self.files)-1, -1, -1):
            (cur_dir, cur_base) = os.path.split (self.files[i])
            if (cur_dir == dir and
                (pattern_re is None or pattern_re.match (cur_base))):
                self.debug_print("removing %s" % self.files[i])
                del self.files[i]
                files_found = 1
    
        return files_found

# class FileList


# ----------------------------------------------------------------------
# Utility functions

def findall (dir = os.curdir):
    """Find all files under 'dir' and return the list of full filenames
    (relative to 'dir').
    """
    from stat import ST_MODE, S_ISREG, S_ISDIR, S_ISLNK

    list = []
    stack = [dir]
    pop = stack.pop
    push = stack.append

    while stack:
        dir = pop()
        names = os.listdir (dir)

        for name in names:
            if dir != os.curdir:        # avoid the dreaded "./" syndrome
                fullname = os.path.join (dir, name)
            else:
                fullname = name

            # Avoid excess stat calls -- just one will do, thank you!
            stat = os.stat(fullname)
            mode = stat[ST_MODE]
            if S_ISREG(mode):
                list.append (fullname)
            elif S_ISDIR(mode) and not S_ISLNK(mode):
                push (fullname)

    return list


def glob_to_re (pattern):
    """Translate a shell-like glob pattern to a regular expression; return
    a string containing the regex.  Differs from 'fnmatch.translate()' in
    that '*' does not match "special characters" (which are
    platform-specific).
    """
    pattern_re = fnmatch.translate (pattern)

    # '?' and '*' in the glob pattern become '.' and '.*' in the RE, which
    # IMHO is wrong -- '?' and '*' aren't supposed to match slash in Unix,
    # and by extension they shouldn't match such "special characters" under
    # any OS.  So change all non-escaped dots in the RE to match any
    # character except the special characters.
    # XXX currently the "special characters" are just slash -- i.e. this is
    # Unix-only.
    pattern_re = re.sub (r'(^|[^\\])\.', r'\1[^/]', pattern_re)
    return pattern_re

# glob_to_re ()


def translate_pattern (pattern, anchor=1, prefix=None, is_regex=0):
    """Translate a shell-like wildcard pattern to a compiled regular
    expression.  Return the compiled regex.  If 'is_regex' true,
    then 'pattern' is directly compiled to a regex (if it's a string)
    or just returned as-is (assumes it's a regex object).
    """
    if is_regex:
        if type(pattern) is StringType:
            return re.compile(pattern)
        else:
            return pattern

    if pattern:
        pattern_re = glob_to_re (pattern)
    else:
        pattern_re = ''
        
    if prefix is not None:
        prefix_re = (glob_to_re (prefix))[0:-1] # ditch trailing $
        pattern_re = "^" + os.path.join (prefix_re, ".*" + pattern_re)
    else:                               # no prefix -- respect anchor flag
        if anchor:
            pattern_re = "^" + pattern_re
        
    return re.compile (pattern_re)

# translate_pattern ()