| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | """distutils.util
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | General-purpose utility functions used throughout the Distutils | 
					
						
							|  |  |  | (especially in command classes).  Mostly filesystem manipulation, but | 
					
						
							|  |  |  | not limited to that.  The functions in this module generally raise | 
					
						
							|  |  |  | DistutilsFileError when they have problems with the filesystem, because | 
					
						
							|  |  |  | os.error in pre-1.5.2 Python only gives the error message and not the | 
					
						
							|  |  |  | file causing it."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # created 1999/03/08, Greg Ward | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | __rcsid__ = "$Id$" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | from distutils.errors import * | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-21 18:37:51 +00:00
										 |  |  | # cache for by mkpath() -- in addition to cheapening redundant calls, | 
					
						
							|  |  |  | # eliminates redundant "creating /foo/bar/baz" messages in dry-run mode | 
					
						
							|  |  |  | PATH_CREATED = {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | # I don't use os.makedirs because a) it's new to Python 1.5.2, and | 
					
						
							|  |  |  | # b) it blows up if the directory already exists (I want to silently | 
					
						
							|  |  |  | # succeed in that case). | 
					
						
							| 
									
										
										
										
											1999-04-04 02:54:20 +00:00
										 |  |  | def mkpath (name, mode=0777, verbose=0, dry_run=0): | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |     """Create a directory and any missing ancestor directories.  If the
 | 
					
						
							|  |  |  |        directory already exists, return silently.  Raise | 
					
						
							|  |  |  |        DistutilsFileError if unable to create some directory along the | 
					
						
							|  |  |  |        way (eg. some sub-path exists, but is a file rather than a | 
					
						
							|  |  |  |        directory).  If 'verbose' is true, print a one-line summary of | 
					
						
							|  |  |  |        each mkdir to stdout."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-21 18:37:51 +00:00
										 |  |  |     global PATH_CREATED | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |     # XXX what's the better way to handle verbosity? print as we create | 
					
						
							|  |  |  |     # each directory in the path (the current behaviour), or only announce | 
					
						
							| 
									
										
										
										
											1999-09-21 18:37:51 +00:00
										 |  |  |     # the creation of the whole path? (quite easy to do the latter since | 
					
						
							|  |  |  |     # we're not using a recursive algorithm) | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-10-03 20:50:41 +00:00
										 |  |  |     name = os.path.normpath (name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |     if os.path.isdir (name): | 
					
						
							|  |  |  |         return | 
					
						
							| 
									
										
										
										
											1999-09-21 18:37:51 +00:00
										 |  |  |     if PATH_CREATED.get (name): | 
					
						
							|  |  |  |         return | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     (head, tail) = os.path.split (name) | 
					
						
							|  |  |  |     tails = [tail]                      # stack of lone dirs to create | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     while head and tail and not os.path.isdir (head): | 
					
						
							|  |  |  |         #print "splitting '%s': " % head, | 
					
						
							|  |  |  |         (head, tail) = os.path.split (head) | 
					
						
							|  |  |  |         #print "to ('%s','%s')" % (head, tail) | 
					
						
							|  |  |  |         tails.insert (0, tail)          # push next higher dir onto stack | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     #print "stack of tails:", tails | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-04-04 02:54:20 +00:00
										 |  |  |     # now 'head' contains the deepest directory that already exists | 
					
						
							|  |  |  |     # (that is, the child of 'head' in 'name' is the highest directory | 
					
						
							|  |  |  |     # that does *not* exist) | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |     for d in tails: | 
					
						
							|  |  |  |         #print "head = %s, d = %s: " % (head, d), | 
					
						
							|  |  |  |         head = os.path.join (head, d) | 
					
						
							| 
									
										
										
										
											1999-09-29 12:14:16 +00:00
										 |  |  |         if PATH_CREATED.get (head): | 
					
						
							|  |  |  |             continue | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |         if verbose: | 
					
						
							|  |  |  |             print "creating", head | 
					
						
							| 
									
										
										
										
											1999-04-04 02:54:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if not dry_run: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 os.mkdir (head) | 
					
						
							|  |  |  |             except os.error, (errno, errstr): | 
					
						
							|  |  |  |                 raise DistutilsFileError, "%s: %s" % (head, errstr) | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-21 18:37:51 +00:00
										 |  |  |         PATH_CREATED[head] = 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | # mkpath () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  | def newer (source, target): | 
					
						
							|  |  |  |     """Return true if 'source' exists and is more recently modified than
 | 
					
						
							|  |  |  |        'target', or if 'source' exists and 'target' doesn't.  Return | 
					
						
							|  |  |  |        false if both exist and 'target' is the same age or younger than | 
					
						
							|  |  |  |        'source'.  Raise DistutilsFileError if 'source' does not | 
					
						
							|  |  |  |        exist."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if not os.path.exists (source): | 
					
						
							|  |  |  |         raise DistutilsFileError, "file '%s' does not exist" % source | 
					
						
							|  |  |  |     if not os.path.exists (target): | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |         return 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  |     from stat import ST_MTIME | 
					
						
							|  |  |  |     mtime1 = os.stat(source)[ST_MTIME] | 
					
						
							|  |  |  |     mtime2 = os.stat(target)[ST_MTIME] | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return mtime1 > mtime2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # newer () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  | def newer_pairwise (sources, targets): | 
					
						
							|  |  |  |     """Walk two filename lists in parallel, testing if each 'target' is
 | 
					
						
							|  |  |  |        up-to-date relative to its corresponding 'source'.  If so, both | 
					
						
							|  |  |  |        are deleted from their respective lists.  Return a list of tuples | 
					
						
							|  |  |  |        containing the deleted (source,target) pairs."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if len (sources) != len (targets): | 
					
						
							|  |  |  |         raise ValueError, "'sources' and 'targets' must be same length" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     goners = [] | 
					
						
							|  |  |  |     for i in range (len (sources)-1, -1, -1): | 
					
						
							|  |  |  |         if not newer (sources[i], targets[i]): | 
					
						
							|  |  |  |             goners.append ((sources[i], targets[i])) | 
					
						
							|  |  |  |             del sources[i] | 
					
						
							|  |  |  |             del targets[i] | 
					
						
							|  |  |  |     goners.reverse() | 
					
						
							|  |  |  |     return goners | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # newer_pairwise () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-01-09 22:48:59 +00:00
										 |  |  | def newer_group (sources, target, missing='error'): | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  |     """Return true if 'target' is out-of-date with respect to any
 | 
					
						
							|  |  |  |        file listed in 'sources'.  In other words, if 'target' exists and | 
					
						
							|  |  |  |        is newer than every file in 'sources', return false; otherwise | 
					
						
							| 
									
										
										
										
											2000-01-09 22:48:59 +00:00
										 |  |  |        return true.  'missing' controls what we do when a source file is | 
					
						
							|  |  |  |        missing; the default ("error") is to blow up with an OSError from | 
					
						
							|  |  |  |        inside 'stat()'; if it is "ignore", we silently drop any missing | 
					
						
							|  |  |  |        source files; if it is "newer", any missing source files make us | 
					
						
							|  |  |  |        assume that 'target' is out-of-date (this is handy in "dry-run" | 
					
						
							|  |  |  |        mode: it'll make you pretend to carry out commands that wouldn't | 
					
						
							|  |  |  |        work because inputs are missing, but that doesn't matter because | 
					
						
							|  |  |  |        you're not actually going to run the commands).""" | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # If the target doesn't even exist, then it's definitely out-of-date. | 
					
						
							|  |  |  |     if not os.path.exists (target): | 
					
						
							|  |  |  |         return 1 | 
					
						
							|  |  |  |     | 
					
						
							|  |  |  |     # Otherwise we have to find out the hard way: if *any* source file | 
					
						
							|  |  |  |     # is more recent than 'target', then 'target' is out-of-date and | 
					
						
							|  |  |  |     # we can immediately return true.  If we fall through to the end | 
					
						
							|  |  |  |     # of the loop, then 'target' is up-to-date and we return false. | 
					
						
							|  |  |  |     from stat import ST_MTIME | 
					
						
							|  |  |  |     target_mtime = os.stat (target)[ST_MTIME] | 
					
						
							|  |  |  |     for source in sources: | 
					
						
							| 
									
										
										
										
											2000-01-09 22:48:59 +00:00
										 |  |  |         if not os.path.exists (source): | 
					
						
							|  |  |  |             if missing == 'error':      # blow up when we stat() the file | 
					
						
							|  |  |  |                 pass                     | 
					
						
							|  |  |  |             elif missing == 'ignore':   # missing source dropped from  | 
					
						
							|  |  |  |                 continue                #  target's dependency list | 
					
						
							|  |  |  |             elif missing == 'newer':    # missing source means target is | 
					
						
							|  |  |  |                 return 1                #  out-of-date | 
					
						
							|  |  |  |              | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  |         source_mtime = os.stat(source)[ST_MTIME] | 
					
						
							|  |  |  |         if source_mtime > target_mtime: | 
					
						
							|  |  |  |             return 1 | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         return 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # newer_group () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-10-03 20:50:41 +00:00
										 |  |  | # XXX this isn't used anywhere, and worse, it has the same name as a method | 
					
						
							|  |  |  | # in Command with subtly different semantics.  (This one just has one | 
					
						
							|  |  |  | # source -> one dest; that one has many sources -> one dest.)  Nuke it? | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | def make_file (src, dst, func, args, | 
					
						
							|  |  |  |                verbose=0, update_message=None, noupdate_message=None): | 
					
						
							|  |  |  |     """Makes 'dst' from 'src' (both filenames) by calling 'func' with
 | 
					
						
							|  |  |  |        'args', but only if it needs to: i.e. if 'dst' does not exist or | 
					
						
							|  |  |  |        'src' is newer than 'dst'."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if newer (src, dst): | 
					
						
							|  |  |  |         if verbose and update_message: | 
					
						
							|  |  |  |             print update_message | 
					
						
							|  |  |  |         apply (func, args) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         if verbose and noupdate_message: | 
					
						
							|  |  |  |             print noupdate_message | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # make_file () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _copy_file_contents (src, dst, buffer_size=16*1024): | 
					
						
							|  |  |  |     """Copy the file 'src' to 'dst'; both must be filenames.  Any error
 | 
					
						
							|  |  |  |        opening either file, reading from 'src', or writing to 'dst', | 
					
						
							|  |  |  |        raises DistutilsFileError.  Data is read/written in chunks of | 
					
						
							|  |  |  |        'buffer_size' bytes (default 16k).  No attempt is made to handle | 
					
						
							|  |  |  |        anything apart from regular files."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Stolen from shutil module in the standard library, but with | 
					
						
							|  |  |  |     # custom error-handling added. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fsrc = None | 
					
						
							|  |  |  |     fdst = None | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             fsrc = open(src, 'rb') | 
					
						
							|  |  |  |         except os.error, (errno, errstr): | 
					
						
							|  |  |  |             raise DistutilsFileError, "could not open %s: %s" % (src, errstr) | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             fdst = open(dst, 'wb') | 
					
						
							|  |  |  |         except os.error, (errno, errstr): | 
					
						
							|  |  |  |             raise DistutilsFileError, "could not create %s: %s" % (dst, errstr) | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         while 1: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 buf = fsrc.read (buffer_size) | 
					
						
							|  |  |  |             except os.error, (errno, errstr): | 
					
						
							|  |  |  |                 raise DistutilsFileError, \ | 
					
						
							|  |  |  |                       "could not read from %s: %s" % (src, errstr) | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |             if not buf: | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 fdst.write(buf) | 
					
						
							|  |  |  |             except os.error, (errno, errstr): | 
					
						
							|  |  |  |                 raise DistutilsFileError, \ | 
					
						
							|  |  |  |                       "could not write to %s: %s" % (dst, errstr) | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         if fdst: | 
					
						
							|  |  |  |             fdst.close() | 
					
						
							|  |  |  |         if fsrc: | 
					
						
							|  |  |  |             fsrc.close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # _copy_file_contents() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def copy_file (src, dst, | 
					
						
							|  |  |  |                preserve_mode=1, | 
					
						
							|  |  |  |                preserve_times=1, | 
					
						
							|  |  |  |                update=0, | 
					
						
							| 
									
										
										
										
											1999-04-04 02:54:20 +00:00
										 |  |  |                verbose=0, | 
					
						
							|  |  |  |                dry_run=0): | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     """Copy a file 'src' to 'dst'.  If 'dst' is a directory, then 'src'
 | 
					
						
							|  |  |  |        is copied there with the same name; otherwise, it must be a | 
					
						
							|  |  |  |        filename.  (If the file exists, it will be ruthlessly clobbered.) | 
					
						
							|  |  |  |        If 'preserve_mode' is true (the default), the file's mode (type | 
					
						
							|  |  |  |        and permission bits, or whatever is analogous on the current | 
					
						
							|  |  |  |        platform) is copied.  If 'preserve_times' is true (the default), | 
					
						
							|  |  |  |        the last-modified and last-access times are copied as well.  If | 
					
						
							|  |  |  |        'update' is true, 'src' will only be copied if 'dst' does not | 
					
						
							|  |  |  |        exist, or if 'dst' does exist but is older than 'src'.  If | 
					
						
							|  |  |  |        'verbose' is true, then a one-line summary of the copy will be | 
					
						
							| 
									
										
										
										
											1999-05-02 21:42:05 +00:00
										 |  |  |        printed to stdout. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        Return true if the file was copied (or would have been copied), | 
					
						
							|  |  |  |        false otherwise (ie. 'update' was true and the destination is | 
					
						
							|  |  |  |        up-to-date)."""
 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # XXX doesn't copy Mac-specific metadata | 
					
						
							|  |  |  |         | 
					
						
							|  |  |  |     from stat import * | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if not os.path.isfile (src): | 
					
						
							|  |  |  |         raise DistutilsFileError, \ | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  |               "can't copy %s: not a regular file" % src | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if os.path.isdir (dst): | 
					
						
							|  |  |  |         dir = dst | 
					
						
							|  |  |  |         dst = os.path.join (dst, os.path.basename (src)) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         dir = os.path.dirname (dst) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if update and not newer (src, dst): | 
					
						
							| 
									
										
										
										
											1999-05-02 21:42:05 +00:00
										 |  |  |         if verbose: | 
					
						
							|  |  |  |             print "not copying %s (output up-to-date)" % src | 
					
						
							|  |  |  |         return 0 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if verbose: | 
					
						
							|  |  |  |         print "copying %s -> %s" % (src, dir) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-04-04 02:54:20 +00:00
										 |  |  |     if dry_run: | 
					
						
							| 
									
										
										
										
											1999-05-02 21:42:05 +00:00
										 |  |  |         return 1 | 
					
						
							| 
									
										
										
										
											1999-04-04 02:54:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     _copy_file_contents (src, dst) | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |     if preserve_mode or preserve_times: | 
					
						
							|  |  |  |         st = os.stat (src) | 
					
						
							| 
									
										
										
										
											1999-06-08 17:05:21 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # According to David Ascher <da@ski.org>, utime() should be done | 
					
						
							|  |  |  |         # before chmod() (at least under NT). | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |         if preserve_times: | 
					
						
							|  |  |  |             os.utime (dst, (st[ST_ATIME], st[ST_MTIME])) | 
					
						
							| 
									
										
										
										
											1999-06-08 17:05:21 +00:00
										 |  |  |         if preserve_mode: | 
					
						
							|  |  |  |             os.chmod (dst, S_IMODE (st[ST_MODE])) | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-05-02 21:42:05 +00:00
										 |  |  |     return 1 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | # copy_file () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def copy_tree (src, dst, | 
					
						
							|  |  |  |                preserve_mode=1, | 
					
						
							|  |  |  |                preserve_times=1, | 
					
						
							|  |  |  |                preserve_symlinks=0, | 
					
						
							|  |  |  |                update=0, | 
					
						
							| 
									
										
										
										
											1999-04-04 02:54:20 +00:00
										 |  |  |                verbose=0, | 
					
						
							|  |  |  |                dry_run=0): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     """Copy an entire directory tree 'src' to a new location 'dst'.  Both
 | 
					
						
							|  |  |  |        'src' and 'dst' must be directory names.  If 'src' is not a | 
					
						
							|  |  |  |        directory, raise DistutilsFileError.  If 'dst' does not exist, it | 
					
						
							| 
									
										
										
										
											1999-10-03 20:50:41 +00:00
										 |  |  |        is created with 'mkpath()'.  The end result of the copy is that | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |        every file in 'src' is copied to 'dst', and directories under | 
					
						
							| 
									
										
										
										
											1999-05-02 21:42:05 +00:00
										 |  |  |        'src' are recursively copied to 'dst'.  Return the list of files | 
					
						
							|  |  |  |        copied (under their output names) -- note that if 'update' is true, | 
					
						
							|  |  |  |        this might be less than the list of files considered.  Return | 
					
						
							|  |  |  |        value is not affected by 'dry_run'. | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |        'preserve_mode' and 'preserve_times' are the same as for | 
					
						
							|  |  |  |        'copy_file'; note that they only apply to regular files, not to | 
					
						
							|  |  |  |        directories.  If 'preserve_symlinks' is true, symlinks will be | 
					
						
							|  |  |  |        copied as symlinks (on platforms that support them!); otherwise | 
					
						
							|  |  |  |        (the default), the destination of the symlink will be copied. | 
					
						
							|  |  |  |        'update' and 'verbose' are the same as for 'copy_file'."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  |     if not dry_run and not os.path.isdir (src): | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |         raise DistutilsFileError, \ | 
					
						
							|  |  |  |               "cannot copy tree %s: not a directory" % src     | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         names = os.listdir (src) | 
					
						
							|  |  |  |     except os.error, (errno, errstr): | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  |         if dry_run: | 
					
						
							|  |  |  |             names = [] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             raise DistutilsFileError, \ | 
					
						
							|  |  |  |                   "error listing files in %s: %s" % (src, errstr) | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-04-04 02:54:20 +00:00
										 |  |  |     if not dry_run: | 
					
						
							|  |  |  |         mkpath (dst, verbose=verbose) | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-05-02 21:42:05 +00:00
										 |  |  |     outputs = [] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |     for n in names: | 
					
						
							|  |  |  |         src_name = os.path.join (src, n) | 
					
						
							|  |  |  |         dst_name = os.path.join (dst, n) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if preserve_symlinks and os.path.islink (src_name): | 
					
						
							|  |  |  |             link_dest = os.readlink (src_name) | 
					
						
							| 
									
										
										
										
											1999-04-04 02:54:20 +00:00
										 |  |  |             if verbose: | 
					
						
							|  |  |  |                 print "linking %s -> %s" % (dst_name, link_dest) | 
					
						
							|  |  |  |             if not dry_run: | 
					
						
							|  |  |  |                 os.symlink (link_dest, dst_name) | 
					
						
							| 
									
										
										
										
											1999-05-02 21:42:05 +00:00
										 |  |  |             outputs.append (dst_name) | 
					
						
							|  |  |  |              | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |         elif os.path.isdir (src_name): | 
					
						
							| 
									
										
										
										
											2000-01-30 19:57:48 +00:00
										 |  |  |             outputs.extend ( | 
					
						
							| 
									
										
										
										
											1999-05-02 21:42:05 +00:00
										 |  |  |                 copy_tree (src_name, dst_name, | 
					
						
							|  |  |  |                            preserve_mode, preserve_times, preserve_symlinks, | 
					
						
							| 
									
										
										
										
											2000-01-30 19:57:48 +00:00
										 |  |  |                            update, verbose, dry_run)) | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											1999-05-02 21:42:05 +00:00
										 |  |  |             if (copy_file (src_name, dst_name, | 
					
						
							|  |  |  |                            preserve_mode, preserve_times, | 
					
						
							|  |  |  |                            update, verbose, dry_run)): | 
					
						
							|  |  |  |                 outputs.append (dst_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return outputs | 
					
						
							| 
									
										
										
										
											1999-03-22 14:52:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | # copy_tree () | 
					
						
							| 
									
										
										
										
											1999-09-13 03:09:38 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # XXX I suspect this is Unix-specific -- need porting help! | 
					
						
							|  |  |  | def move_file (src, dst, | 
					
						
							|  |  |  |                verbose=0, | 
					
						
							|  |  |  |                dry_run=0): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """Move a file 'src' to 'dst'.  If 'dst' is a directory, the file
 | 
					
						
							|  |  |  |        will be moved into it with the same name; otherwise, 'src' is | 
					
						
							|  |  |  |        just renamed to 'dst'.  Return the new full name of the file. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |        Handles cross-device moves on Unix using | 
					
						
							|  |  |  |        'copy_file()'.  What about other systems???"""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     from os.path import exists, isfile, isdir, basename, dirname | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if verbose: | 
					
						
							|  |  |  |         print "moving %s -> %s" % (src, dst) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if dry_run: | 
					
						
							|  |  |  |         return dst | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if not isfile (src): | 
					
						
							|  |  |  |         raise DistutilsFileError, \ | 
					
						
							|  |  |  |               "can't move '%s': not a regular file" % src | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if isdir (dst): | 
					
						
							|  |  |  |         dst = os.path.join (dst, basename (src)) | 
					
						
							|  |  |  |     elif exists (dst): | 
					
						
							|  |  |  |         raise DistutilsFileError, \ | 
					
						
							|  |  |  |               "can't move '%s': destination '%s' already exists" % \ | 
					
						
							|  |  |  |               (src, dst) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if not isdir (dirname (dst)): | 
					
						
							|  |  |  |         raise DistutilsFileError, \ | 
					
						
							|  |  |  |               "can't move '%s': destination '%s' not a valid path" % \ | 
					
						
							|  |  |  |               (src, dst) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     copy_it = 0 | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         os.rename (src, dst) | 
					
						
							|  |  |  |     except os.error, (num, msg): | 
					
						
							|  |  |  |         if num == errno.EXDEV: | 
					
						
							|  |  |  |             copy_it = 1 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             raise DistutilsFileError, \ | 
					
						
							|  |  |  |                   "couldn't move '%s' to '%s': %s" % (src, dst, msg) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if copy_it: | 
					
						
							|  |  |  |         copy_file (src, dst) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             os.unlink (src) | 
					
						
							|  |  |  |         except os.error, (num, msg): | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 os.unlink (dst) | 
					
						
							|  |  |  |             except os.error: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |             raise DistutilsFileError, \ | 
					
						
							|  |  |  |                   ("couldn't move '%s' to '%s' by copy/delete: " +  | 
					
						
							|  |  |  |                    "delete '%s' failed: %s") % \ | 
					
						
							|  |  |  |                   (src, dst, src, msg) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return dst | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # move_file () | 
					
						
							| 
									
										
										
										
											1999-09-21 18:37:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def write_file (filename, contents): | 
					
						
							| 
									
										
										
										
											1999-10-03 20:50:41 +00:00
										 |  |  |     """Create a file with the specified name and write 'contents' (a
 | 
					
						
							| 
									
										
										
										
											1999-09-21 18:37:51 +00:00
										 |  |  |        sequence of strings without line terminators) to it."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     f = open (filename, "w") | 
					
						
							|  |  |  |     for line in contents: | 
					
						
							|  |  |  |         f.write (line + "\n") | 
					
						
							|  |  |  |     f.close () |