shutil.py 38.2 KB
Newer Older
1
"""Utility functions for copying and archiving files and directory trees.
2

3
XXX The functions here don't copy the resource fork or other metadata on Mac.
4 5

"""
Guido van Rossum's avatar
Guido van Rossum committed
6

Guido van Rossum's avatar
Guido van Rossum committed
7
import os
8
import sys
9
import stat
10
from os.path import abspath
Georg Brandl's avatar
Georg Brandl committed
11
import fnmatch
12
import collections
13
import errno
14
import tarfile
15

16 17
try:
    import bz2
18
    del bz2
19
    _BZ2_SUPPORTED = True
20
except ImportError:
21 22
    _BZ2_SUPPORTED = False

23 24
try:
    from pwd import getpwnam
25
except ImportError:
26 27 28 29
    getpwnam = None

try:
    from grp import getgrnam
30
except ImportError:
31
    getgrnam = None
Guido van Rossum's avatar
Guido van Rossum committed
32

33 34 35
__all__ = ["copyfileobj", "copyfile", "copymode", "copystat", "copy", "copy2",
           "copytree", "move", "rmtree", "Error", "SpecialFileError",
           "ExecError", "make_archive", "get_archive_formats",
36 37
           "register_archive_format", "unregister_archive_format",
           "get_unpack_formats", "register_unpack_format",
Éric Araujo's avatar
Éric Araujo committed
38
           "unregister_unpack_format", "unpack_archive",
39
           "ignore_patterns", "chown", "which"]
40
           # disk_usage is added later, if available on the platform
41

42
class Error(OSError):
43
    pass
Guido van Rossum's avatar
Guido van Rossum committed
44

45 46 47
class SameFileError(Error):
    """Raised when source and destination are the same file."""

48
class SpecialFileError(OSError):
49 50 51
    """Raised when trying to do a kind of operation (e.g. copying) which is
    not supported on a special file (e.g. a named pipe)"""

52
class ExecError(OSError):
53 54
    """Raised when a command could not be executed"""

55
class ReadError(OSError):
56 57 58
    """Raised when an archive cannot be read"""

class RegistryError(Exception):
59
    """Raised when a registry operation with the archiving
60 61 62
    and unpacking registeries fails"""


63 64 65 66 67 68 69 70
def copyfileobj(fsrc, fdst, length=16*1024):
    """copy data from file-like object fsrc to file-like object fdst"""
    while 1:
        buf = fsrc.read(length)
        if not buf:
            break
        fdst.write(buf)

71 72
def _samefile(src, dst):
    # Macintosh, Unix.
73
    if hasattr(os.path, 'samefile'):
74 75 76 77
        try:
            return os.path.samefile(src, dst)
        except OSError:
            return False
78 79 80 81

    # All other platforms: check for same pathname.
    return (os.path.normcase(os.path.abspath(src)) ==
            os.path.normcase(os.path.abspath(dst)))
Tim Peters's avatar
Tim Peters committed
82

83
def copyfile(src, dst, *, follow_symlinks=True):
84 85
    """Copy data from src to dst.

86
    If follow_symlinks is not set and src is a symbolic link, a new
87 88 89
    symlink will be created instead of copying the file it points to.

    """
90
    if _samefile(src, dst):
91
        raise SameFileError("{!r} and {!r} are the same file".format(src, dst))
92

93 94 95 96 97 98
    for fn in [src, dst]:
        try:
            st = os.stat(fn)
        except OSError:
            # File most likely does not exist
            pass
99 100 101 102
        else:
            # XXX What about other special files? (sockets, devices...)
            if stat.S_ISFIFO(st.st_mode):
                raise SpecialFileError("`%s` is a named pipe" % fn)
103

104
    if not follow_symlinks and os.path.islink(src):
105 106 107 108 109
        os.symlink(os.readlink(src), dst)
    else:
        with open(src, 'rb') as fsrc:
            with open(dst, 'wb') as fdst:
                copyfileobj(fsrc, fdst)
110
    return dst
111

112
def copymode(src, dst, *, follow_symlinks=True):
113
    """Copy mode bits from src to dst.
Guido van Rossum's avatar
Guido van Rossum committed
114

115 116 117
    If follow_symlinks is not set, symlinks aren't followed if and only
    if both `src` and `dst` are symlinks.  If `lchmod` isn't available
    (e.g. Linux) this method does nothing.
118 119

    """
120
    if not follow_symlinks and os.path.islink(src) and os.path.islink(dst):
121 122 123 124 125 126 127 128
        if hasattr(os, 'lchmod'):
            stat_func, chmod_func = os.lstat, os.lchmod
        else:
            return
    elif hasattr(os, 'chmod'):
        stat_func, chmod_func = os.stat, os.chmod
    else:
        return
Guido van Rossum's avatar
Guido van Rossum committed
129

130 131 132
    st = stat_func(src)
    chmod_func(dst, stat.S_IMODE(st.st_mode))

133
if hasattr(os, 'listxattr'):
134
    def _copyxattr(src, dst, *, follow_symlinks=True):
135 136 137 138
        """Copy extended filesystem attributes from `src` to `dst`.

        Overwrite existing attributes.

139
        If `follow_symlinks` is false, symlinks won't be followed.
140 141 142

        """

143 144 145 146 147 148 149
        try:
            names = os.listxattr(src, follow_symlinks=follow_symlinks)
        except OSError as e:
            if e.errno not in (errno.ENOTSUP, errno.ENODATA):
                raise
            return
        for name in names:
150
            try:
151 152
                value = os.getxattr(src, name, follow_symlinks=follow_symlinks)
                os.setxattr(dst, name, value, follow_symlinks=follow_symlinks)
153 154 155 156 157 158 159
            except OSError as e:
                if e.errno not in (errno.EPERM, errno.ENOTSUP, errno.ENODATA):
                    raise
else:
    def _copyxattr(*args, **kwargs):
        pass

160
def copystat(src, dst, *, follow_symlinks=True):
161 162
    """Copy all stat info (mode bits, atime, mtime, flags) from src to dst.

163
    If the optional flag `follow_symlinks` is not set, symlinks aren't followed if and
164 165 166
    only if both `src` and `dst` are symlinks.

    """
167
    def _nop(*args, ns=None, follow_symlinks=None):
168 169
        pass

170
    # follow symlinks (aka don't not follow symlinks)
171
    follow = follow_symlinks or not (os.path.islink(src) and os.path.islink(dst))
172 173 174 175
    if follow:
        # use the real function if it exists
        def lookup(name):
            return getattr(os, name, _nop)
176
    else:
177 178 179 180 181 182 183 184 185
        # use the real function only if it exists
        # *and* it supports follow_symlinks
        def lookup(name):
            fn = getattr(os, name, _nop)
            if fn in os.supports_follow_symlinks:
                return fn
            return _nop

    st = lookup("stat")(src, follow_symlinks=follow)
186
    mode = stat.S_IMODE(st.st_mode)
187 188 189 190 191 192 193 194 195
    lookup("utime")(dst, ns=(st.st_atime_ns, st.st_mtime_ns),
        follow_symlinks=follow)
    try:
        lookup("chmod")(dst, mode, follow_symlinks=follow)
    except NotImplementedError:
        # if we got a NotImplementedError, it's because
        #   * follow_symlinks=False,
        #   * lchown() is unavailable, and
        #   * either
196
        #       * fchownat() is unavailable or
197 198 199 200 201 202
        #       * fchownat() doesn't implement AT_SYMLINK_NOFOLLOW.
        #         (it returned ENOSUP.)
        # therefore we're out of options--we simply cannot chown the
        # symlink.  give up, suppress the error.
        # (which is what shutil always did in this circumstance.)
        pass
203
    if hasattr(st, 'st_flags'):
204
        try:
205
            lookup("chflags")(dst, st.st_flags, follow_symlinks=follow)
206
        except OSError as why:
207 208 209 210
            for err in 'EOPNOTSUPP', 'ENOTSUP':
                if hasattr(errno, err) and why.errno == getattr(errno, err):
                    break
            else:
211
                raise
212
    _copyxattr(src, dst, follow_symlinks=follow)
213

214
def copy(src, dst, *, follow_symlinks=True):
215
    """Copy data and mode bits ("cp src dst"). Return the file's destination.
Tim Peters's avatar
Tim Peters committed
216

217 218
    The destination may be a directory.

219
    If follow_symlinks is false, symlinks won't be followed. This
220 221
    resembles GNU's "cp -P src dst".

222 223 224
    If source and destination are the same file, a SameFileError will be
    raised.

225
    """
226
    if os.path.isdir(dst):
227
        dst = os.path.join(dst, os.path.basename(src))
228 229
    copyfile(src, dst, follow_symlinks=follow_symlinks)
    copymode(src, dst, follow_symlinks=follow_symlinks)
230
    return dst
Guido van Rossum's avatar
Guido van Rossum committed
231

232
def copy2(src, dst, *, follow_symlinks=True):
233 234
    """Copy data and all stat info ("cp -p src dst"). Return the file's
    destination."
235 236 237

    The destination may be a directory.

238
    If follow_symlinks is false, symlinks won't be followed. This
239 240
    resembles GNU's "cp -P src dst".

241
    """
242
    if os.path.isdir(dst):
243
        dst = os.path.join(dst, os.path.basename(src))
244 245
    copyfile(src, dst, follow_symlinks=follow_symlinks)
    copystat(src, dst, follow_symlinks=follow_symlinks)
246
    return dst
Guido van Rossum's avatar
Guido van Rossum committed
247

Georg Brandl's avatar
Georg Brandl committed
248 249
def ignore_patterns(*patterns):
    """Function that can be used as copytree() ignore parameter.
250

Georg Brandl's avatar
Georg Brandl committed
251 252 253 254 255 256 257 258 259
    Patterns is a sequence of glob-style patterns
    that are used to exclude files"""
    def _ignore_patterns(path, names):
        ignored_names = []
        for pattern in patterns:
            ignored_names.extend(fnmatch.filter(names, pattern))
        return set(ignored_names)
    return _ignore_patterns

260 261
def copytree(src, dst, symlinks=False, ignore=None, copy_function=copy2,
             ignore_dangling_symlinks=False):
262
    """Recursively copy a directory tree.
263 264

    The destination directory must not already exist.
265
    If exception(s) occur, an Error is raised with a list of reasons.
266 267 268 269

    If the optional symlinks flag is true, symbolic links in the
    source tree result in symbolic links in the destination tree; if
    it is false, the contents of the files pointed to by symbolic
270 271 272 273 274
    links are copied. If the file pointed by the symlink doesn't
    exist, an exception will be added in the list of errors raised in
    an Error exception at the end of the copy process.

    You can set the optional ignore_dangling_symlinks flag to true if you
275 276
    want to silence this exception. Notice that this has no effect on
    platforms that don't support os.symlink.
277

Georg Brandl's avatar
Georg Brandl committed
278 279 280 281 282 283 284 285 286 287 288 289
    The optional ignore argument is a callable. If given, it
    is called with the `src` parameter, which is the directory
    being visited by copytree(), and `names` which is the list of
    `src` contents, as returned by os.listdir():

        callable(src, names) -> ignored_names

    Since copytree() is called recursively, the callable will be
    called once for each directory that is copied. It returns a
    list of names relative to the `src` directory that should
    not be copied.

290 291 292 293
    The optional copy_function argument is a callable that will be used
    to copy each file. It will be called with the source path and the
    destination path as arguments. By default, copy2() is used, but any
    function that supports the same signature (like copy()) can be used.
294 295

    """
296
    names = os.listdir(src)
Georg Brandl's avatar
Georg Brandl committed
297 298 299 300 301
    if ignore is not None:
        ignored_names = ignore(src, names)
    else:
        ignored_names = set()

Johannes Gijsbers's avatar
Johannes Gijsbers committed
302
    os.makedirs(dst)
303
    errors = []
304
    for name in names:
Georg Brandl's avatar
Georg Brandl committed
305 306
        if name in ignored_names:
            continue
307 308 309
        srcname = os.path.join(src, name)
        dstname = os.path.join(dst, name)
        try:
310
            if os.path.islink(srcname):
311
                linkto = os.readlink(srcname)
312
                if symlinks:
313 314 315
                    # We can't just leave it to `copy_function` because legacy
                    # code with a custom `copy_function` may rely on copytree
                    # doing the right thing.
316
                    os.symlink(linkto, dstname)
317
                    copystat(srcname, dstname, follow_symlinks=not symlinks)
318 319 320 321 322 323
                else:
                    # ignore dangling symlink if the flag is on
                    if not os.path.exists(linkto) and ignore_dangling_symlinks:
                        continue
                    # otherwise let the copy occurs. copy2 will raise an error
                    copy_function(srcname, dstname)
324
            elif os.path.isdir(srcname):
325
                copytree(srcname, dstname, symlinks, ignore, copy_function)
326
            else:
327
                # Will raise a SpecialFileError for unsupported file types
328
                copy_function(srcname, dstname)
329 330
        # catch the Error from the recursive copytree so that we can
        # continue with other files
331
        except Error as err:
332
            errors.extend(err.args[0])
333
        except OSError as why:
334
            errors.append((srcname, dstname, str(why)))
335 336
    try:
        copystat(src, dst)
337
    except OSError as why:
338 339
        # Copying file access times may fail on Windows
        if why.winerror is None:
340
            errors.append((src, dst, str(why)))
341
    if errors:
342
        raise Error(errors)
343
    return dst
344

345 346
# version vulnerable to race conditions
def _rmtree_unsafe(path, onerror):
347 348 349 350 351 352 353 354
    try:
        if os.path.islink(path):
            # symlinks to directories are forbidden, see bug #1669
            raise OSError("Cannot call rmtree on a symbolic link")
    except OSError:
        onerror(os.path.islink, path, sys.exc_info())
        # can't continue even if onerror hook returns
        return
355 356 357
    names = []
    try:
        names = os.listdir(path)
358
    except OSError:
359 360 361 362 363
        onerror(os.listdir, path, sys.exc_info())
    for name in names:
        fullname = os.path.join(path, name)
        try:
            mode = os.lstat(fullname).st_mode
364
        except OSError:
365 366
            mode = 0
        if stat.S_ISDIR(mode):
367
            _rmtree_unsafe(fullname, onerror)
368
        else:
369
            try:
370
                os.unlink(fullname)
371
            except OSError:
372
                onerror(os.unlink, fullname, sys.exc_info())
373 374
    try:
        os.rmdir(path)
375
    except OSError:
376
        onerror(os.rmdir, path, sys.exc_info())
377

378 379 380 381
# Version using fd-based APIs to protect against races
def _rmtree_safe_fd(topfd, path, onerror):
    names = []
    try:
382
        names = os.listdir(topfd)
383 384
    except OSError as err:
        err.filename = path
385
        onerror(os.listdir, path, sys.exc_info())
386 387 388
    for name in names:
        fullname = os.path.join(path, name)
        try:
389
            orig_st = os.stat(name, dir_fd=topfd, follow_symlinks=False)
390
            mode = orig_st.st_mode
391
        except OSError:
392 393 394
            mode = 0
        if stat.S_ISDIR(mode):
            try:
395
                dirfd = os.open(name, os.O_RDONLY, dir_fd=topfd)
396
            except OSError:
397
                onerror(os.open, fullname, sys.exc_info())
398 399 400 401
            else:
                try:
                    if os.path.samestat(orig_st, os.fstat(dirfd)):
                        _rmtree_safe_fd(dirfd, fullname, onerror)
402 403
                        try:
                            os.rmdir(name, dir_fd=topfd)
404
                        except OSError:
405
                            onerror(os.rmdir, fullname, sys.exc_info())
406 407 408 409 410 411 412 413 414
                    else:
                        try:
                            # This can only happen if someone replaces
                            # a directory with a symlink after the call to
                            # stat.S_ISDIR above.
                            raise OSError("Cannot call rmtree on a symbolic "
                                          "link")
                        except OSError:
                            onerror(os.path.islink, fullname, sys.exc_info())
415 416 417 418
                finally:
                    os.close(dirfd)
        else:
            try:
419
                os.unlink(name, dir_fd=topfd)
420
            except OSError:
421
                onerror(os.unlink, fullname, sys.exc_info())
422

423 424 425 426
_use_fd_functions = ({os.open, os.stat, os.unlink, os.rmdir} <=
                     os.supports_dir_fd and
                     os.listdir in os.supports_fd and
                     os.stat in os.supports_follow_symlinks)
427

428 429 430 431 432
def rmtree(path, ignore_errors=False, onerror=None):
    """Recursively delete a directory tree.

    If ignore_errors is set, errors are ignored; otherwise, if onerror
    is set, it is called to handle the error with arguments (func,
433
    path, exc_info) where func is platform and implementation dependent;
434 435 436 437 438 439 440 441 442 443 444 445
    path is the argument to that function that caused it to fail; and
    exc_info is a tuple returned by sys.exc_info().  If ignore_errors
    is false and onerror is None, an exception is raised.

    """
    if ignore_errors:
        def onerror(*args):
            pass
    elif onerror is None:
        def onerror(*args):
            raise
    if _use_fd_functions:
446 447 448
        # While the unsafe rmtree works fine on bytes, the fd based does not.
        if isinstance(path, bytes):
            path = os.fsdecode(path)
449 450 451 452 453 454 455 456 457 458 459 460 461
        # Note: To guard against symlink races, we use the standard
        # lstat()/open()/fstat() trick.
        try:
            orig_st = os.lstat(path)
        except Exception:
            onerror(os.lstat, path, sys.exc_info())
            return
        try:
            fd = os.open(path, os.O_RDONLY)
        except Exception:
            onerror(os.lstat, path, sys.exc_info())
            return
        try:
462
            if os.path.samestat(orig_st, os.fstat(fd)):
463
                _rmtree_safe_fd(fd, path, onerror)
464 465
                try:
                    os.rmdir(path)
466
                except OSError:
467
                    onerror(os.rmdir, path, sys.exc_info())
468
            else:
469 470 471 472 473
                try:
                    # symlinks to directories are forbidden, see bug #1669
                    raise OSError("Cannot call rmtree on a symbolic link")
                except OSError:
                    onerror(os.path.islink, path, sys.exc_info())
474 475 476 477 478
        finally:
            os.close(fd)
    else:
        return _rmtree_unsafe(path, onerror)

479 480 481
# Allow introspection of whether or not the hardening against symlink
# attacks is supported on the current platform
rmtree.avoids_symlink_attacks = _use_fd_functions
482 483 484 485 486 487

def _basename(path):
    # A basename() variant which first strips the trailing slash, if present.
    # Thus we always get the last component of the path, even for directories.
    return os.path.basename(path.rstrip(os.path.sep))

488
def move(src, dst):
489
    """Recursively move a file or directory to another location. This is
490 491
    similar to the Unix "mv" command. Return the file or directory's
    destination.
492 493 494 495

    If the destination is a directory or a symlink to a directory, the source
    is moved inside the directory. The destination path must not already
    exist.
496

497 498 499 500
    If the destination already exists but is not a directory, it may be
    overwritten depending on os.rename() semantics.

    If the destination is on our current filesystem, then rename() is used.
501 502 503 504
    Otherwise, src is copied to the destination and then removed. Symlinks are
    recreated under the new name if os.rename() fails because of cross
    filesystem renames.

505 506 507 508
    A lot more could be done here...  A look at a mv.c shows a lot of
    the issues this implementation glosses over.

    """
509 510
    real_dst = dst
    if os.path.isdir(dst):
511 512 513 514 515 516
        if _samefile(src, dst):
            # We might be on a case insensitive filesystem,
            # perform the rename anyway.
            os.rename(src, dst)
            return

517 518 519
        real_dst = os.path.join(dst, _basename(src))
        if os.path.exists(real_dst):
            raise Error("Destination path '%s' already exists" % real_dst)
520
    try:
521
        os.rename(src, real_dst)
522
    except OSError:
523 524 525 526 527
        if os.path.islink(src):
            linkto = os.readlink(src)
            os.symlink(linkto, real_dst)
            os.unlink(src)
        elif os.path.isdir(src):
528
            if _destinsrc(src, dst):
529
                raise Error("Cannot move a directory '%s' into itself '%s'." % (src, dst))
530
            copytree(src, real_dst, symlinks=True)
531 532
            rmtree(src)
        else:
533
            copy2(src, real_dst)
534
            os.unlink(src)
535
    return real_dst
536

537
def _destinsrc(src, dst):
538 539 540 541 542 543 544
    src = abspath(src)
    dst = abspath(dst)
    if not src.endswith(os.path.sep):
        src += os.path.sep
    if not dst.endswith(os.path.sep):
        dst += os.path.sep
    return dst.startswith(src)
545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574

def _get_gid(name):
    """Returns a gid, given a group name."""
    if getgrnam is None or name is None:
        return None
    try:
        result = getgrnam(name)
    except KeyError:
        result = None
    if result is not None:
        return result[2]
    return None

def _get_uid(name):
    """Returns an uid, given a user name."""
    if getpwnam is None or name is None:
        return None
    try:
        result = getpwnam(name)
    except KeyError:
        result = None
    if result is not None:
        return result[2]
    return None

def _make_tarball(base_name, base_dir, compress="gzip", verbose=0, dry_run=0,
                  owner=None, group=None, logger=None):
    """Create a (possibly compressed) tar file from all the files under
    'base_dir'.

575
    'compress' must be "gzip" (the default), "bzip2", or None.
576 577 578 579 580

    'owner' and 'group' can be used to define an owner and a group for the
    archive that is being built. If not provided, the current owner and group
    will be used.

581
    The output tar file will be named 'base_name' +  ".tar", possibly plus
582
    the appropriate compression extension (".gz", or ".bz2").
583 584 585

    Returns the output filename.
    """
586 587 588 589 590 591
    tar_compression = {'gzip': 'gz', None: ''}
    compress_ext = {'gzip': '.gz'}

    if _BZ2_SUPPORTED:
        tar_compression['bzip2'] = 'bz2'
        compress_ext['bzip2'] = '.bz2'
592 593

    # flags for compression program, each element of list will be an argument
594
    if compress is not None and compress not in compress_ext:
595 596
        raise ValueError("bad value for 'compress', or compression format not "
                         "supported : {0}".format(compress))
597

598
    archive_name = base_name + '.tar' + compress_ext.get(compress, '')
599
    archive_dir = os.path.dirname(archive_name)
600

601
    if not os.path.exists(archive_dir):
602
        if logger is not None:
Éric Araujo's avatar
Éric Araujo committed
603
            logger.info("creating %s", archive_dir)
604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631
        if not dry_run:
            os.makedirs(archive_dir)

    # creating the tarball
    if logger is not None:
        logger.info('Creating tar archive')

    uid = _get_uid(owner)
    gid = _get_gid(group)

    def _set_uid_gid(tarinfo):
        if gid is not None:
            tarinfo.gid = gid
            tarinfo.gname = group
        if uid is not None:
            tarinfo.uid = uid
            tarinfo.uname = owner
        return tarinfo

    if not dry_run:
        tar = tarfile.open(archive_name, 'w|%s' % tar_compression[compress])
        try:
            tar.add(base_dir, filter=_set_uid_gid)
        finally:
            tar.close()

    return archive_name

632
def _call_external_zip(base_dir, zip_filename, verbose=False, dry_run=False):
633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651
    # XXX see if we want to keep an external call here
    if verbose:
        zipoptions = "-r"
    else:
        zipoptions = "-rq"
    from distutils.errors import DistutilsExecError
    from distutils.spawn import spawn
    try:
        spawn(["zip", zipoptions, zip_filename, base_dir], dry_run=dry_run)
    except DistutilsExecError:
        # XXX really should distinguish between "couldn't find
        # external 'zip' command" and "zip failed".
        raise ExecError("unable to create zip file '%s': "
            "could neither import the 'zipfile' module nor "
            "find a standalone zip utility") % zip_filename

def _make_zipfile(base_name, base_dir, verbose=0, dry_run=0, logger=None):
    """Create a zip file from all the files under 'base_dir'.

652
    The output zip file will be named 'base_name' + ".zip".  Uses either the
653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670
    "zipfile" Python module (if available) or the InfoZIP "zip" utility
    (if installed and found on the default search path).  If neither tool is
    available, raises ExecError.  Returns the name of the output zip
    file.
    """
    zip_filename = base_name + ".zip"
    archive_dir = os.path.dirname(base_name)

    if not os.path.exists(archive_dir):
        if logger is not None:
            logger.info("creating %s", archive_dir)
        if not dry_run:
            os.makedirs(archive_dir)

    # If zipfile module is not available, try spawning an external 'zip'
    # command.
    try:
        import zipfile
671
    except ImportError:
672 673 674
        zipfile = None

    if zipfile is None:
675
        _call_external_zip(base_dir, zip_filename, verbose, dry_run)
676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698
    else:
        if logger is not None:
            logger.info("creating '%s' and adding '%s' to it",
                        zip_filename, base_dir)

        if not dry_run:
            zip = zipfile.ZipFile(zip_filename, "w",
                                  compression=zipfile.ZIP_DEFLATED)

            for dirpath, dirnames, filenames in os.walk(base_dir):
                for name in filenames:
                    path = os.path.normpath(os.path.join(dirpath, name))
                    if os.path.isfile(path):
                        zip.write(path, path)
                        if logger is not None:
                            logger.info("adding '%s'", path)
            zip.close()

    return zip_filename

_ARCHIVE_FORMATS = {
    'gztar': (_make_tarball, [('compress', 'gzip')], "gzip'ed tar-file"),
    'tar':   (_make_tarball, [('compress', None)], "uncompressed tar file"),
699
    'zip':   (_make_zipfile, [], "ZIP file")
700 701
    }

702 703 704 705
if _BZ2_SUPPORTED:
    _ARCHIVE_FORMATS['bztar'] = (_make_tarball, [('compress', 'bzip2')],
                                "bzip2'ed tar-file")

706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726
def get_archive_formats():
    """Returns a list of supported formats for archiving and unarchiving.

    Each element of the returned sequence is a tuple (name, description)
    """
    formats = [(name, registry[2]) for name, registry in
               _ARCHIVE_FORMATS.items()]
    formats.sort()
    return formats

def register_archive_format(name, function, extra_args=None, description=''):
    """Registers an archive format.

    name is the name of the format. function is the callable that will be
    used to create archives. If provided, extra_args is a sequence of
    (name, value) tuples that will be passed as arguments to the callable.
    description can be provided to describe the format, and will be returned
    by the get_archive_formats() function.
    """
    if extra_args is None:
        extra_args = []
727
    if not callable(function):
728 729 730 731
        raise TypeError('The %s object is not callable' % function)
    if not isinstance(extra_args, (tuple, list)):
        raise TypeError('extra_args needs to be a sequence')
    for element in extra_args:
732
        if not isinstance(element, (tuple, list)) or len(element) !=2:
733 734 735 736 737 738 739 740 741 742 743 744
            raise TypeError('extra_args elements are : (arg_name, value)')

    _ARCHIVE_FORMATS[name] = (function, extra_args, description)

def unregister_archive_format(name):
    del _ARCHIVE_FORMATS[name]

def make_archive(base_name, format, root_dir=None, base_dir=None, verbose=0,
                 dry_run=0, owner=None, group=None, logger=None):
    """Create an archive file (eg. zip or tar).

    'base_name' is the name of the file to create, minus any format-specific
745 746
    extension; 'format' is the archive format: one of "zip", "tar", "bztar"
    or "gztar".
747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792

    'root_dir' is a directory that will be the root directory of the
    archive; ie. we typically chdir into 'root_dir' before creating the
    archive.  'base_dir' is the directory where we start archiving from;
    ie. 'base_dir' will be the common prefix of all files and
    directories in the archive.  'root_dir' and 'base_dir' both default
    to the current directory.  Returns the name of the archive file.

    'owner' and 'group' are used when creating a tar archive. By default,
    uses the current owner and group.
    """
    save_cwd = os.getcwd()
    if root_dir is not None:
        if logger is not None:
            logger.debug("changing into '%s'", root_dir)
        base_name = os.path.abspath(base_name)
        if not dry_run:
            os.chdir(root_dir)

    if base_dir is None:
        base_dir = os.curdir

    kwargs = {'dry_run': dry_run, 'logger': logger}

    try:
        format_info = _ARCHIVE_FORMATS[format]
    except KeyError:
        raise ValueError("unknown archive format '%s'" % format)

    func = format_info[0]
    for arg, val in format_info[1]:
        kwargs[arg] = val

    if format != 'zip':
        kwargs['owner'] = owner
        kwargs['group'] = group

    try:
        filename = func(base_name, base_dir, **kwargs)
    finally:
        if root_dir is not None:
            if logger is not None:
                logger.debug("changing back to '%s'", save_cwd)
            os.chdir(save_cwd)

    return filename
793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819


def get_unpack_formats():
    """Returns a list of supported formats for unpacking.

    Each element of the returned sequence is a tuple
    (name, extensions, description)
    """
    formats = [(name, info[0], info[3]) for name, info in
               _UNPACK_FORMATS.items()]
    formats.sort()
    return formats

def _check_unpack_options(extensions, function, extra_args):
    """Checks what gets registered as an unpacker."""
    # first make sure no other unpacker is registered for this extension
    existing_extensions = {}
    for name, info in _UNPACK_FORMATS.items():
        for ext in info[0]:
            existing_extensions[ext] = name

    for extension in extensions:
        if extension in existing_extensions:
            msg = '%s is already registered for "%s"'
            raise RegistryError(msg % (extension,
                                       existing_extensions[extension]))

820
    if not callable(function):
821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860
        raise TypeError('The registered function must be a callable')


def register_unpack_format(name, extensions, function, extra_args=None,
                           description=''):
    """Registers an unpack format.

    `name` is the name of the format. `extensions` is a list of extensions
    corresponding to the format.

    `function` is the callable that will be
    used to unpack archives. The callable will receive archives to unpack.
    If it's unable to handle an archive, it needs to raise a ReadError
    exception.

    If provided, `extra_args` is a sequence of
    (name, value) tuples that will be passed as arguments to the callable.
    description can be provided to describe the format, and will be returned
    by the get_unpack_formats() function.
    """
    if extra_args is None:
        extra_args = []
    _check_unpack_options(extensions, function, extra_args)
    _UNPACK_FORMATS[name] = extensions, function, extra_args, description

def unregister_unpack_format(name):
    """Removes the pack format from the registery."""
    del _UNPACK_FORMATS[name]

def _ensure_directory(path):
    """Ensure that the parent directory of `path` exists"""
    dirname = os.path.dirname(path)
    if not os.path.isdir(dirname):
        os.makedirs(dirname)

def _unpack_zipfile(filename, extract_dir):
    """Unpack zip `filename` to `extract_dir`
    """
    try:
        import zipfile
861
    except ImportError:
862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883
        raise ReadError('zlib not supported, cannot unpack this archive.')

    if not zipfile.is_zipfile(filename):
        raise ReadError("%s is not a zip file" % filename)

    zip = zipfile.ZipFile(filename)
    try:
        for info in zip.infolist():
            name = info.filename

            # don't extract absolute paths or ones with .. in them
            if name.startswith('/') or '..' in name:
                continue

            target = os.path.join(extract_dir, *name.split('/'))
            if not target:
                continue

            _ensure_directory(target)
            if not name.endswith('/'):
                # file
                data = zip.read(info.filename)
884
                f = open(target, 'wb')
885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911
                try:
                    f.write(data)
                finally:
                    f.close()
                    del data
    finally:
        zip.close()

def _unpack_tarfile(filename, extract_dir):
    """Unpack tar/tar.gz/tar.bz2 `filename` to `extract_dir`
    """
    try:
        tarobj = tarfile.open(filename)
    except tarfile.TarError:
        raise ReadError(
            "%s is not a compressed or uncompressed tar file" % filename)
    try:
        tarobj.extractall(extract_dir)
    finally:
        tarobj.close()

_UNPACK_FORMATS = {
    'gztar': (['.tar.gz', '.tgz'], _unpack_tarfile, [], "gzip'ed tar-file"),
    'tar':   (['.tar'], _unpack_tarfile, [], "uncompressed tar file"),
    'zip':   (['.zip'], _unpack_zipfile, [], "ZIP file")
    }

912 913 914 915
if _BZ2_SUPPORTED:
    _UNPACK_FORMATS['bztar'] = (['.bz2'], _unpack_tarfile, [],
                                "bzip2'ed tar-file")

916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946
def _find_unpack_format(filename):
    for name, info in _UNPACK_FORMATS.items():
        for extension in info[0]:
            if filename.endswith(extension):
                return name
    return None

def unpack_archive(filename, extract_dir=None, format=None):
    """Unpack an archive.

    `filename` is the name of the archive.

    `extract_dir` is the name of the target directory, where the archive
    is unpacked. If not provided, the current working directory is used.

    `format` is the archive format: one of "zip", "tar", or "gztar". Or any
    other registered format. If not provided, unpack_archive will use the
    filename extension and see if an unpacker was registered for that
    extension.

    In case none is found, a ValueError is raised.
    """
    if extract_dir is None:
        extract_dir = os.getcwd()

    if format is not None:
        try:
            format_info = _UNPACK_FORMATS[format]
        except KeyError:
            raise ValueError("Unknown unpack format '{0}'".format(format))

947 948
        func = format_info[1]
        func(filename, extract_dir, **dict(format_info[2]))
949 950 951 952 953 954 955 956 957
    else:
        # we need to look at the registered unpackers supported extensions
        format = _find_unpack_format(filename)
        if format is None:
            raise ReadError("Unknown archive format '{0}'".format(filename))

        func = _UNPACK_FORMATS[format][1]
        kwargs = dict(_UNPACK_FORMATS[format][2])
        func(filename, extract_dir, **kwargs)
958

959 960 961 962 963

if hasattr(os, 'statvfs'):

    __all__.append('disk_usage')
    _ntuple_diskusage = collections.namedtuple('usage', 'total used free')
964 965

    def disk_usage(path):
966 967
        """Return disk usage statistics about the given path.

968
        Returned value is a named tuple with attributes 'total', 'used' and
969
        'free', which are the amount of total, used and free space, in bytes.
970
        """
971 972 973 974 975 976 977 978 979 980 981 982 983 984 985
        st = os.statvfs(path)
        free = st.f_bavail * st.f_frsize
        total = st.f_blocks * st.f_frsize
        used = (st.f_blocks - st.f_bfree) * st.f_frsize
        return _ntuple_diskusage(total, used, free)

elif os.name == 'nt':

    import nt
    __all__.append('disk_usage')
    _ntuple_diskusage = collections.namedtuple('usage', 'total used free')

    def disk_usage(path):
        """Return disk usage statistics about the given path.

986
        Returned values is a named tuple with attributes 'total', 'used' and
987 988 989 990
        'free', which are the amount of total, used and free space, in bytes.
        """
        total, free = nt._getdiskusage(path)
        used = total - free
991
        return _ntuple_diskusage(total, used, free)
992

993

994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023
def chown(path, user=None, group=None):
    """Change owner user and group of the given path.

    user and group can be the uid/gid or the user/group names, and in that case,
    they are converted to their respective uid/gid.
    """

    if user is None and group is None:
        raise ValueError("user and/or group must be set")

    _user = user
    _group = group

    # -1 means don't change it
    if user is None:
        _user = -1
    # user can either be an int (the uid) or a string (the system username)
    elif isinstance(user, str):
        _user = _get_uid(user)
        if _user is None:
            raise LookupError("no such user: {!r}".format(user))

    if group is None:
        _group = -1
    elif not isinstance(group, int):
        _group = _get_gid(group)
        if _group is None:
            raise LookupError("no such group: {!r}".format(group))

    os.chown(path, _user, _group)
1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066

def get_terminal_size(fallback=(80, 24)):
    """Get the size of the terminal window.

    For each of the two dimensions, the environment variable, COLUMNS
    and LINES respectively, is checked. If the variable is defined and
    the value is a positive integer, it is used.

    When COLUMNS or LINES is not defined, which is the common case,
    the terminal connected to sys.__stdout__ is queried
    by invoking os.get_terminal_size.

    If the terminal size cannot be successfully queried, either because
    the system doesn't support querying, or because we are not
    connected to a terminal, the value given in fallback parameter
    is used. Fallback defaults to (80, 24) which is the default
    size used by many terminal emulators.

    The value returned is a named tuple of type os.terminal_size.
    """
    # columns, lines are the working values
    try:
        columns = int(os.environ['COLUMNS'])
    except (KeyError, ValueError):
        columns = 0

    try:
        lines = int(os.environ['LINES'])
    except (KeyError, ValueError):
        lines = 0

    # only query if necessary
    if columns <= 0 or lines <= 0:
        try:
            size = os.get_terminal_size(sys.__stdout__.fileno())
        except (NameError, OSError):
            size = os.terminal_size(fallback)
        if columns <= 0:
            columns = size.columns
        if lines <= 0:
            lines = size.lines

    return os.terminal_size((columns, lines))
1067 1068

def which(cmd, mode=os.F_OK | os.X_OK, path=None):
1069
    """Given a command, mode, and a PATH string, return the path which
1070 1071 1072 1073 1074 1075 1076 1077
    conforms to the given mode on the PATH, or None if there is no such
    file.

    `mode` defaults to os.F_OK | os.X_OK. `path` defaults to the result
    of os.environ.get("PATH"), or can be overridden with a custom search
    path.

    """
1078 1079 1080 1081
    # Check that a given file can be accessed with the correct mode.
    # Additionally check that `file` is not a directory, as on Windows
    # directories pass the os.access check.
    def _access_check(fn, mode):
1082 1083
        return (os.path.exists(fn) and os.access(fn, mode)
                and not os.path.isdir(fn))
1084

1085 1086 1087 1088 1089 1090 1091
    # If we're given a path with a directory part, look it up directly rather
    # than referring to PATH directories. This includes checking relative to the
    # current directory, e.g. ./script
    if os.path.dirname(cmd):
        if _access_check(cmd, mode):
            return cmd
        return None
1092

1093 1094 1095 1096 1097
    if path is None:
        path = os.environ.get("PATH", os.defpath)
    if not path:
        return None
    path = path.split(os.pathsep)
1098 1099 1100 1101 1102 1103 1104 1105 1106 1107

    if sys.platform == "win32":
        # The current directory takes precedence on Windows.
        if not os.curdir in path:
            path.insert(0, os.curdir)

        # PATHEXT is necessary to check on Windows.
        pathext = os.environ.get("PATHEXT", "").split(os.pathsep)
        # See if the given file matches any of the expected path extensions.
        # This will allow us to short circuit when given "python.exe".
1108 1109
        # If it does match, only test that one, otherwise we have to try
        # others.
1110 1111 1112 1113
        if any(cmd.lower().endswith(ext.lower()) for ext in pathext):
            files = [cmd]
        else:
            files = [cmd + ext for ext in pathext]
1114 1115 1116 1117 1118 1119 1120
    else:
        # On other platforms you don't have things like PATHEXT to tell you
        # what file suffixes are executable, so just pass on cmd as-is.
        files = [cmd]

    seen = set()
    for dir in path:
1121 1122 1123
        normdir = os.path.normcase(dir)
        if not normdir in seen:
            seen.add(normdir)
1124 1125 1126 1127 1128
            for thefile in files:
                name = os.path.join(dir, thefile)
                if _access_check(name, mode):
                    return name
    return None