shutil.py 45.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
Georg Brandl's avatar
Georg Brandl committed
10
import fnmatch
11
import collections
12
import errno
13
import io
14 15 16 17 18 19 20

try:
    import zlib
    del zlib
    _ZLIB_SUPPORTED = True
except ImportError:
    _ZLIB_SUPPORTED = False
21

22 23
try:
    import bz2
24
    del bz2
25
    _BZ2_SUPPORTED = True
26
except ImportError:
27 28
    _BZ2_SUPPORTED = False

29 30 31 32 33 34 35
try:
    import lzma
    del lzma
    _LZMA_SUPPORTED = True
except ImportError:
    _LZMA_SUPPORTED = False

36 37
try:
    from pwd import getpwnam
38
except ImportError:
39 40 41 42
    getpwnam = None

try:
    from grp import getgrnam
43
except ImportError:
44
    getgrnam = None
Guido van Rossum's avatar
Guido van Rossum committed
45

46
_WINDOWS = os.name == 'nt'
47 48 49
posix = nt = None
if os.name == 'posix':
    import posix
50
elif _WINDOWS:
51 52
    import nt

53
COPY_BUFSIZE = 1024 * 1024 if _WINDOWS else 16 * 1024
54
_HAS_SENDFILE = posix and hasattr(os, "sendfile")
55
_HAS_FCOPYFILE = posix and hasattr(posix, "_fcopyfile")  # macOS
56

57 58 59
__all__ = ["copyfileobj", "copyfile", "copymode", "copystat", "copy", "copy2",
           "copytree", "move", "rmtree", "Error", "SpecialFileError",
           "ExecError", "make_archive", "get_archive_formats",
60 61
           "register_archive_format", "unregister_archive_format",
           "get_unpack_formats", "register_unpack_format",
Éric Araujo's avatar
Éric Araujo committed
62
           "unregister_unpack_format", "unpack_archive",
63 64
           "ignore_patterns", "chown", "which", "get_terminal_size",
           "SameFileError"]
65
           # disk_usage is added later, if available on the platform
66

67
class Error(OSError):
68
    pass
Guido van Rossum's avatar
Guido van Rossum committed
69

70 71 72
class SameFileError(Error):
    """Raised when source and destination are the same file."""

73
class SpecialFileError(OSError):
74 75 76
    """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)"""

77
class ExecError(OSError):
78 79
    """Raised when a command could not be executed"""

80
class ReadError(OSError):
81 82 83
    """Raised when an archive cannot be read"""

class RegistryError(Exception):
84
    """Raised when a registry operation with the archiving
85
    and unpacking registries fails"""
86

87 88 89 90 91
class _GiveupOnFastCopy(Exception):
    """Raised as a signal to fallback on using raw read()/write()
    file copy when fast-copy functions fail to do so.
    """

92
def _fastcopy_fcopyfile(fsrc, fdst, flags):
93
    """Copy a regular file content or metadata by using high-performance
94
    fcopyfile(3) syscall (macOS).
95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171
    """
    try:
        infd = fsrc.fileno()
        outfd = fdst.fileno()
    except Exception as err:
        raise _GiveupOnFastCopy(err)  # not a regular file

    try:
        posix._fcopyfile(infd, outfd, flags)
    except OSError as err:
        err.filename = fsrc.name
        err.filename2 = fdst.name
        if err.errno in {errno.EINVAL, errno.ENOTSUP}:
            raise _GiveupOnFastCopy(err)
        else:
            raise err from None

def _fastcopy_sendfile(fsrc, fdst):
    """Copy data from one regular mmap-like fd to another by using
    high-performance sendfile(2) syscall.
    This should work on Linux >= 2.6.33 and Solaris only.
    """
    # Note: copyfileobj() is left alone in order to not introduce any
    # unexpected breakage. Possible risks by using zero-copy calls
    # in copyfileobj() are:
    # - fdst cannot be open in "a"(ppend) mode
    # - fsrc and fdst may be open in "t"(ext) mode
    # - fsrc may be a BufferedReader (which hides unread data in a buffer),
    #   GzipFile (which decompresses data), HTTPResponse (which decodes
    #   chunks).
    # - possibly others (e.g. encrypted fs/partition?)
    global _HAS_SENDFILE
    try:
        infd = fsrc.fileno()
        outfd = fdst.fileno()
    except Exception as err:
        raise _GiveupOnFastCopy(err)  # not a regular file

    # Hopefully the whole file will be copied in a single call.
    # sendfile() is called in a loop 'till EOF is reached (0 return)
    # so a bufsize smaller or bigger than the actual file size
    # should not make any difference, also in case the file content
    # changes while being copied.
    try:
        blocksize = max(os.fstat(infd).st_size, 2 ** 23)  # min 8MB
    except Exception:
        blocksize = 2 ** 27  # 128MB

    offset = 0
    while True:
        try:
            sent = os.sendfile(outfd, infd, offset, blocksize)
        except OSError as err:
            # ...in oder to have a more informative exception.
            err.filename = fsrc.name
            err.filename2 = fdst.name

            if err.errno == errno.ENOTSOCK:
                # sendfile() on this platform (probably Linux < 2.6.33)
                # does not support copies between regular files (only
                # sockets).
                _HAS_SENDFILE = False
                raise _GiveupOnFastCopy(err)

            if err.errno == errno.ENOSPC:  # filesystem is full
                raise err from None

            # Give up on first call and if no data was copied.
            if offset == 0 and os.lseek(outfd, 0, os.SEEK_CUR) == 0:
                raise _GiveupOnFastCopy(err)

            raise err
        else:
            if sent == 0:
                break  # EOF
            offset += sent

172 173 174 175 176
def _copyfileobj_readinto(fsrc, fdst, length=COPY_BUFSIZE):
    """readinto()/memoryview() based variant of copyfileobj().
    *fsrc* must support readinto() method and both files must be
    open in binary mode.
    """
177 178 179 180 181 182 183 184 185
    # Localize variable access to minimize overhead.
    fsrc_readinto = fsrc.readinto
    fdst_write = fdst.write
    with memoryview(bytearray(length)) as mv:
        while True:
            n = fsrc_readinto(mv)
            if not n:
                break
            elif n < length:
186 187
                with mv[:n] as smv:
                    fdst.write(smv)
188 189 190 191
            else:
                fdst_write(mv)

def copyfileobj(fsrc, fdst, length=COPY_BUFSIZE):
192
    """copy data from file-like object fsrc to file-like object fdst"""
193 194 195 196 197 198 199 200
    # Localize variable access to minimize overhead.
    fsrc_read = fsrc.read
    fdst_write = fdst.write
    while True:
        buf = fsrc_read(length)
        if not buf:
            break
        fdst_write(buf)
201

202 203
def _samefile(src, dst):
    # Macintosh, Unix.
204
    if hasattr(os.path, 'samefile'):
205 206 207 208
        try:
            return os.path.samefile(src, dst)
        except OSError:
            return False
209 210 211 212

    # 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
213

214
def copyfile(src, dst, *, follow_symlinks=True):
215
    """Copy data from src to dst in the most efficient way possible.
216

217
    If follow_symlinks is not set and src is a symbolic link, a new
218 219 220
    symlink will be created instead of copying the file it points to.

    """
221
    if _samefile(src, dst):
222
        raise SameFileError("{!r} and {!r} are the same file".format(src, dst))
223

224 225
    file_size = 0
    for i, fn in enumerate([src, dst]):
226 227 228 229 230
        try:
            st = os.stat(fn)
        except OSError:
            # File most likely does not exist
            pass
231 232 233 234
        else:
            # XXX What about other special files? (sockets, devices...)
            if stat.S_ISFIFO(st.st_mode):
                raise SpecialFileError("`%s` is a named pipe" % fn)
235 236
            if _WINDOWS and i == 0:
                file_size = st.st_size
237

238
    if not follow_symlinks and os.path.islink(src):
239 240
        os.symlink(os.readlink(src), dst)
    else:
241
        with open(src, 'rb') as fsrc, open(dst, 'wb') as fdst:
242 243
            # macOS
            if _HAS_FCOPYFILE:
244
                try:
245
                    _fastcopy_fcopyfile(fsrc, fdst, posix._COPYFILE_DATA)
246 247 248
                    return dst
                except _GiveupOnFastCopy:
                    pass
249 250
            # Linux / Solaris
            elif _HAS_SENDFILE:
251
                try:
252
                    _fastcopy_sendfile(fsrc, fdst)
253 254 255
                    return dst
                except _GiveupOnFastCopy:
                    pass
256 257 258 259 260
            # Windows, see:
            # https://github.com/python/cpython/pull/7160#discussion_r195405230
            elif _WINDOWS and file_size > 0:
                _copyfileobj_readinto(fsrc, fdst, min(file_size, COPY_BUFSIZE))
                return dst
261

262
            copyfileobj(fsrc, fdst)
263

264
    return dst
265

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

269 270 271
    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.
272 273

    """
274
    if not follow_symlinks and os.path.islink(src) and os.path.islink(dst):
275 276 277 278 279 280 281 282
        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
283

284 285 286
    st = stat_func(src)
    chmod_func(dst, stat.S_IMODE(st.st_mode))

287
if hasattr(os, 'listxattr'):
288
    def _copyxattr(src, dst, *, follow_symlinks=True):
289 290 291 292
        """Copy extended filesystem attributes from `src` to `dst`.

        Overwrite existing attributes.

293
        If `follow_symlinks` is false, symlinks won't be followed.
294 295 296

        """

297 298 299 300 301 302 303
        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:
304
            try:
305 306
                value = os.getxattr(src, name, follow_symlinks=follow_symlinks)
                os.setxattr(dst, name, value, follow_symlinks=follow_symlinks)
307 308 309 310 311 312 313
            except OSError as e:
                if e.errno not in (errno.EPERM, errno.ENOTSUP, errno.ENODATA):
                    raise
else:
    def _copyxattr(*args, **kwargs):
        pass

314
def copystat(src, dst, *, follow_symlinks=True):
315
    """Copy file metadata
316

317 318 319 320
    Copy the permission bits, last access time, last modification time, and
    flags from `src` to `dst`. On Linux, copystat() also copies the "extended
    attributes" where possible. The file contents, owner, and group are
    unaffected. `src` and `dst` are path names given as strings.
321

322 323
    If the optional flag `follow_symlinks` is not set, symlinks aren't
    followed if and only if both `src` and `dst` are symlinks.
324
    """
325
    def _nop(*args, ns=None, follow_symlinks=None):
326 327
        pass

328
    # follow symlinks (aka don't not follow symlinks)
329
    follow = follow_symlinks or not (os.path.islink(src) and os.path.islink(dst))
330 331 332 333
    if follow:
        # use the real function if it exists
        def lookup(name):
            return getattr(os, name, _nop)
334
    else:
335 336 337 338 339 340 341 342 343
        # 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)
344
    mode = stat.S_IMODE(st.st_mode)
345 346 347 348 349 350 351 352 353
    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
354
        #       * fchownat() is unavailable or
355 356 357 358 359 360
        #       * 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
361
    if hasattr(st, 'st_flags'):
362
        try:
363
            lookup("chflags")(dst, st.st_flags, follow_symlinks=follow)
364
        except OSError as why:
365 366 367 368
            for err in 'EOPNOTSUPP', 'ENOTSUP':
                if hasattr(errno, err) and why.errno == getattr(errno, err):
                    break
            else:
369
                raise
370
    _copyxattr(src, dst, follow_symlinks=follow)
371

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

375 376
    The destination may be a directory.

377
    If follow_symlinks is false, symlinks won't be followed. This
378 379
    resembles GNU's "cp -P src dst".

380 381 382
    If source and destination are the same file, a SameFileError will be
    raised.

383
    """
384
    if os.path.isdir(dst):
385
        dst = os.path.join(dst, os.path.basename(src))
386 387
    copyfile(src, dst, follow_symlinks=follow_symlinks)
    copymode(src, dst, follow_symlinks=follow_symlinks)
388
    return dst
Guido van Rossum's avatar
Guido van Rossum committed
389

390
def copy2(src, dst, *, follow_symlinks=True):
391 392 393 394
    """Copy data and metadata. Return the file's destination.

    Metadata is copied with copystat(). Please see the copystat function
    for more information.
395 396 397

    The destination may be a directory.

398
    If follow_symlinks is false, symlinks won't be followed. This
399
    resembles GNU's "cp -P src dst".
400
    """
401
    if os.path.isdir(dst):
402
        dst = os.path.join(dst, os.path.basename(src))
403 404
    copyfile(src, dst, follow_symlinks=follow_symlinks)
    copystat(src, dst, follow_symlinks=follow_symlinks)
405
    return dst
Guido van Rossum's avatar
Guido van Rossum committed
406

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

Georg Brandl's avatar
Georg Brandl committed
410 411 412 413 414 415 416 417 418
    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

419 420
def copytree(src, dst, symlinks=False, ignore=None, copy_function=copy2,
             ignore_dangling_symlinks=False):
421
    """Recursively copy a directory tree.
422 423

    The destination directory must not already exist.
424
    If exception(s) occur, an Error is raised with a list of reasons.
425 426 427 428

    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
429 430 431 432 433
    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
434 435
    want to silence this exception. Notice that this has no effect on
    platforms that don't support os.symlink.
436

Georg Brandl's avatar
Georg Brandl committed
437 438 439 440 441 442 443 444 445 446 447 448
    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.

449 450 451 452
    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.
453 454

    """
455
    names = os.listdir(src)
Georg Brandl's avatar
Georg Brandl committed
456 457 458 459 460
    if ignore is not None:
        ignored_names = ignore(src, names)
    else:
        ignored_names = set()

Johannes Gijsbers's avatar
Johannes Gijsbers committed
461
    os.makedirs(dst)
462
    errors = []
463
    for name in names:
Georg Brandl's avatar
Georg Brandl committed
464 465
        if name in ignored_names:
            continue
466 467 468
        srcname = os.path.join(src, name)
        dstname = os.path.join(dst, name)
        try:
469
            if os.path.islink(srcname):
470
                linkto = os.readlink(srcname)
471
                if symlinks:
472 473 474
                    # 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.
475
                    os.symlink(linkto, dstname)
476
                    copystat(srcname, dstname, follow_symlinks=not symlinks)
477 478 479 480 481
                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
482 483 484 485 486
                    if os.path.isdir(srcname):
                        copytree(srcname, dstname, symlinks, ignore,
                                 copy_function)
                    else:
                        copy_function(srcname, dstname)
487
            elif os.path.isdir(srcname):
488
                copytree(srcname, dstname, symlinks, ignore, copy_function)
489
            else:
490
                # Will raise a SpecialFileError for unsupported file types
491
                copy_function(srcname, dstname)
492 493
        # catch the Error from the recursive copytree so that we can
        # continue with other files
494
        except Error as err:
495
            errors.extend(err.args[0])
496
        except OSError as why:
497
            errors.append((srcname, dstname, str(why)))
498 499
    try:
        copystat(src, dst)
500
    except OSError as why:
501
        # Copying file access times may fail on Windows
502
        if getattr(why, 'winerror', None) is None:
503
            errors.append((src, dst, str(why)))
504
    if errors:
505
        raise Error(errors)
506
    return dst
507

508 509
# version vulnerable to race conditions
def _rmtree_unsafe(path, onerror):
510
    try:
511 512
        with os.scandir(path) as scandir_it:
            entries = list(scandir_it)
513
    except OSError:
514 515 516 517
        onerror(os.scandir, path, sys.exc_info())
        entries = []
    for entry in entries:
        fullname = entry.path
518
        try:
519
            is_dir = entry.is_dir(follow_symlinks=False)
520
        except OSError:
521 522 523 524 525 526 527 528 529 530 531
            is_dir = False
        if is_dir:
            try:
                if entry.is_symlink():
                    # This can only happen if someone replaces
                    # a directory with a symlink after the call to
                    # os.scandir or entry.is_dir above.
                    raise OSError("Cannot call rmtree on a symbolic link")
            except OSError:
                onerror(os.path.islink, fullname, sys.exc_info())
                continue
532
            _rmtree_unsafe(fullname, onerror)
533
        else:
534
            try:
535
                os.unlink(fullname)
536
            except OSError:
537
                onerror(os.unlink, fullname, sys.exc_info())
538 539
    try:
        os.rmdir(path)
540
    except OSError:
541
        onerror(os.rmdir, path, sys.exc_info())
542

543 544 545
# Version using fd-based APIs to protect against races
def _rmtree_safe_fd(topfd, path, onerror):
    try:
546 547
        with os.scandir(topfd) as scandir_it:
            entries = list(scandir_it)
548 549
    except OSError as err:
        err.filename = path
550 551 552 553
        onerror(os.scandir, path, sys.exc_info())
        return
    for entry in entries:
        fullname = os.path.join(path, entry.name)
554
        try:
555 556 557 558
            is_dir = entry.is_dir(follow_symlinks=False)
            if is_dir:
                orig_st = entry.stat(follow_symlinks=False)
                is_dir = stat.S_ISDIR(orig_st.st_mode)
559
        except OSError:
560 561
            is_dir = False
        if is_dir:
562
            try:
563
                dirfd = os.open(entry.name, os.O_RDONLY, dir_fd=topfd)
564
            except OSError:
565
                onerror(os.open, fullname, sys.exc_info())
566 567 568 569
            else:
                try:
                    if os.path.samestat(orig_st, os.fstat(dirfd)):
                        _rmtree_safe_fd(dirfd, fullname, onerror)
570
                        try:
571
                            os.rmdir(entry.name, dir_fd=topfd)
572
                        except OSError:
573
                            onerror(os.rmdir, fullname, sys.exc_info())
574 575 576 577
                    else:
                        try:
                            # This can only happen if someone replaces
                            # a directory with a symlink after the call to
578
                            # os.scandir or stat.S_ISDIR above.
579 580 581 582
                            raise OSError("Cannot call rmtree on a symbolic "
                                          "link")
                        except OSError:
                            onerror(os.path.islink, fullname, sys.exc_info())
583 584 585 586
                finally:
                    os.close(dirfd)
        else:
            try:
587
                os.unlink(entry.name, dir_fd=topfd)
588
            except OSError:
589
                onerror(os.unlink, fullname, sys.exc_info())
590

591 592
_use_fd_functions = ({os.open, os.stat, os.unlink, os.rmdir} <=
                     os.supports_dir_fd and
593
                     os.scandir in os.supports_fd and
594
                     os.stat in os.supports_follow_symlinks)
595

596 597 598 599 600
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,
601
    path, exc_info) where func is platform and implementation dependent;
602 603 604 605 606 607 608 609 610 611 612 613
    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:
614 615 616
        # While the unsafe rmtree works fine on bytes, the fd based does not.
        if isinstance(path, bytes):
            path = os.fsdecode(path)
617 618 619 620 621 622 623 624 625 626 627 628 629
        # 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:
630
            if os.path.samestat(orig_st, os.fstat(fd)):
631
                _rmtree_safe_fd(fd, path, onerror)
632 633
                try:
                    os.rmdir(path)
634
                except OSError:
635
                    onerror(os.rmdir, path, sys.exc_info())
636
            else:
637 638 639 640 641
                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())
642 643 644
        finally:
            os.close(fd)
    else:
645 646 647 648 649 650 651 652
        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
653 654
        return _rmtree_unsafe(path, onerror)

655 656 657
# Allow introspection of whether or not the hardening against symlink
# attacks is supported on the current platform
rmtree.avoids_symlink_attacks = _use_fd_functions
658 659 660 661

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.
662 663
    sep = os.path.sep + (os.path.altsep or '')
    return os.path.basename(path.rstrip(sep))
664

665
def move(src, dst, copy_function=copy2):
666
    """Recursively move a file or directory to another location. This is
667 668
    similar to the Unix "mv" command. Return the file or directory's
    destination.
669 670 671 672

    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.
673

674 675 676 677
    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.
678 679 680 681
    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.

682 683 684 685 686
    The optional `copy_function` argument is a callable that will be used
    to copy the source or it will be delegated to `copytree`.
    By default, copy2() is used, but any function that supports the same
    signature (like copy()) can be used.

687 688 689 690
    A lot more could be done here...  A look at a mv.c shows a lot of
    the issues this implementation glosses over.

    """
691 692
    real_dst = dst
    if os.path.isdir(dst):
693 694 695 696 697 698
        if _samefile(src, dst):
            # We might be on a case insensitive filesystem,
            # perform the rename anyway.
            os.rename(src, dst)
            return

699 700 701
        real_dst = os.path.join(dst, _basename(src))
        if os.path.exists(real_dst):
            raise Error("Destination path '%s' already exists" % real_dst)
702
    try:
703
        os.rename(src, real_dst)
704
    except OSError:
705 706 707 708 709
        if os.path.islink(src):
            linkto = os.readlink(src)
            os.symlink(linkto, real_dst)
            os.unlink(src)
        elif os.path.isdir(src):
710
            if _destinsrc(src, dst):
711 712 713 714
                raise Error("Cannot move a directory '%s' into itself"
                            " '%s'." % (src, dst))
            copytree(src, real_dst, copy_function=copy_function,
                     symlinks=True)
715 716
            rmtree(src)
        else:
717
            copy_function(src, real_dst)
718
            os.unlink(src)
719
    return real_dst
720

721
def _destinsrc(src, dst):
722 723
    src = os.path.abspath(src)
    dst = os.path.abspath(dst)
724 725 726 727 728
    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)
729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758

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'.

759
    'compress' must be "gzip" (the default), "bzip2", "xz", or None.
760 761 762 763 764

    '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.

765
    The output tar file will be named 'base_name' +  ".tar", possibly plus
766
    the appropriate compression extension (".gz", ".bz2", or ".xz").
767 768 769

    Returns the output filename.
    """
770 771 772 773 774 775 776 777 778
    if compress is None:
        tar_compression = ''
    elif _ZLIB_SUPPORTED and compress == 'gzip':
        tar_compression = 'gz'
    elif _BZ2_SUPPORTED and compress == 'bzip2':
        tar_compression = 'bz2'
    elif _LZMA_SUPPORTED and compress == 'xz':
        tar_compression = 'xz'
    else:
779 780
        raise ValueError("bad value for 'compress', or compression format not "
                         "supported : {0}".format(compress))
781

782 783 784 785
    import tarfile  # late import for breaking circular dependency

    compress_ext = '.' + tar_compression if compress else ''
    archive_name = base_name + '.tar' + compress_ext
786
    archive_dir = os.path.dirname(archive_name)
787

788
    if archive_dir and not os.path.exists(archive_dir):
789
        if logger is not None:
Éric Araujo's avatar
Éric Araujo committed
790
            logger.info("creating %s", archive_dir)
791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810
        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:
811
        tar = tarfile.open(archive_name, 'w|%s' % tar_compression)
812 813 814 815 816 817 818 819 820 821
        try:
            tar.add(base_dir, filter=_set_uid_gid)
        finally:
            tar.close()

    return archive_name

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'.

822 823
    The output zip file will be named 'base_name' + ".zip".  Returns the
    name of the output zip file.
824
    """
825
    import zipfile  # late import for breaking circular dependency
826

827 828 829
    zip_filename = base_name + ".zip"
    archive_dir = os.path.dirname(base_name)

830
    if archive_dir and not os.path.exists(archive_dir):
831 832 833 834 835
        if logger is not None:
            logger.info("creating %s", archive_dir)
        if not dry_run:
            os.makedirs(archive_dir)

836 837 838
    if logger is not None:
        logger.info("creating '%s' and adding '%s' to it",
                    zip_filename, base_dir)
839

840 841 842
    if not dry_run:
        with zipfile.ZipFile(zip_filename, "w",
                             compression=zipfile.ZIP_DEFLATED) as zf:
843
            path = os.path.normpath(base_dir)
844 845 846 847
            if path != os.curdir:
                zf.write(path, path)
                if logger is not None:
                    logger.info("adding '%s'", path)
848
            for dirpath, dirnames, filenames in os.walk(base_dir):
849 850 851 852 853
                for name in sorted(dirnames):
                    path = os.path.normpath(os.path.join(dirpath, name))
                    zf.write(path, path)
                    if logger is not None:
                        logger.info("adding '%s'", path)
854 855 856 857 858 859
                for name in filenames:
                    path = os.path.normpath(os.path.join(dirpath, name))
                    if os.path.isfile(path):
                        zf.write(path, path)
                        if logger is not None:
                            logger.info("adding '%s'", path)
860 861 862 863 864

    return zip_filename

_ARCHIVE_FORMATS = {
    'tar':   (_make_tarball, [('compress', None)], "uncompressed tar file"),
865 866 867 868 869 870
}

if _ZLIB_SUPPORTED:
    _ARCHIVE_FORMATS['gztar'] = (_make_tarball, [('compress', 'gzip')],
                                "gzip'ed tar-file")
    _ARCHIVE_FORMATS['zip'] = (_make_zipfile, [], "ZIP file")
871

872 873 874 875
if _BZ2_SUPPORTED:
    _ARCHIVE_FORMATS['bztar'] = (_make_tarball, [('compress', 'bzip2')],
                                "bzip2'ed tar-file")

876 877 878 879
if _LZMA_SUPPORTED:
    _ARCHIVE_FORMATS['xztar'] = (_make_tarball, [('compress', 'xz')],
                                "xz'ed tar-file")

880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900
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 = []
901
    if not callable(function):
902 903 904 905
        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:
906
        if not isinstance(element, (tuple, list)) or len(element) !=2:
907 908 909 910 911 912 913 914 915 916 917 918
            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
919 920
    extension; 'format' is the archive format: one of "zip", "tar", "gztar",
    "bztar", or "xztar".  Or any other registered format.
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 947

    '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:
948
        raise ValueError("unknown archive format '%s'" % format) from None
949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966

    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
967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993


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]))

994
    if not callable(function):
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
        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):
1021
    """Removes the pack format from the registry."""
1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032
    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`
    """
1033
    import zipfile  # late import for breaking circular dependency
1034 1035 1036 1037 1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054

    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)
1055
                f = open(target, 'wb')
1056 1057 1058 1059 1060 1061 1062 1063 1064
                try:
                    f.write(data)
                finally:
                    f.close()
                    del data
    finally:
        zip.close()

def _unpack_tarfile(filename, extract_dir):
1065
    """Unpack tar/tar.gz/tar.bz2/tar.xz `filename` to `extract_dir`
1066
    """
1067
    import tarfile  # late import for breaking circular dependency
1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079
    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 = {
    'tar':   (['.tar'], _unpack_tarfile, [], "uncompressed tar file"),
1080 1081 1082 1083 1084 1085
    'zip':   (['.zip'], _unpack_zipfile, [], "ZIP file"),
}

if _ZLIB_SUPPORTED:
    _UNPACK_FORMATS['gztar'] = (['.tar.gz', '.tgz'], _unpack_tarfile, [],
                                "gzip'ed tar-file")
1086

1087
if _BZ2_SUPPORTED:
1088
    _UNPACK_FORMATS['bztar'] = (['.tar.bz2', '.tbz2'], _unpack_tarfile, [],
1089 1090
                                "bzip2'ed tar-file")

1091 1092 1093 1094
if _LZMA_SUPPORTED:
    _UNPACK_FORMATS['xztar'] = (['.tar.xz', '.txz'], _unpack_tarfile, [],
                                "xz'ed tar-file")

1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109
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.

1110 1111 1112 1113
    `format` is the archive format: one of "zip", "tar", "gztar", "bztar",
    or "xztar".  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.
1114 1115 1116 1117 1118 1119

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

1120 1121 1122
    extract_dir = os.fspath(extract_dir)
    filename = os.fspath(filename)

1123 1124 1125 1126
    if format is not None:
        try:
            format_info = _UNPACK_FORMATS[format]
        except KeyError:
1127
            raise ValueError("Unknown unpack format '{0}'".format(format)) from None
1128

1129 1130
        func = format_info[1]
        func(filename, extract_dir, **dict(format_info[2]))
1131 1132 1133 1134 1135 1136 1137 1138 1139
    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)
1140

1141 1142 1143 1144 1145

if hasattr(os, 'statvfs'):

    __all__.append('disk_usage')
    _ntuple_diskusage = collections.namedtuple('usage', 'total used free')
1146 1147 1148
    _ntuple_diskusage.total.__doc__ = 'Total space in bytes'
    _ntuple_diskusage.used.__doc__ = 'Used space in bytes'
    _ntuple_diskusage.free.__doc__ = 'Free space in bytes'
1149 1150

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

1153
        Returned value is a named tuple with attributes 'total', 'used' and
1154
        'free', which are the amount of total, used and free space, in bytes.
1155
        """
1156 1157 1158 1159 1160 1161
        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)

1162
elif _WINDOWS:
1163 1164 1165 1166 1167 1168 1169

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

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

1170
        Returned values is a named tuple with attributes 'total', 'used' and
1171 1172 1173 1174
        'free', which are the amount of total, used and free space, in bytes.
        """
        total, free = nt._getdiskusage(path)
        used = total - free
1175
        return _ntuple_diskusage(total, used, free)
1176

1177

1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207
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)
1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242

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())
1243 1244 1245
        except (AttributeError, ValueError, OSError):
            # stdout is None, closed, detached, or not a terminal, or
            # os.get_terminal_size() is unsupported
1246 1247 1248 1249 1250 1251 1252
            size = os.terminal_size(fallback)
        if columns <= 0:
            columns = size.columns
        if lines <= 0:
            lines = size.lines

    return os.terminal_size((columns, lines))
1253 1254

def which(cmd, mode=os.F_OK | os.X_OK, path=None):
1255
    """Given a command, mode, and a PATH string, return the path which
1256 1257 1258 1259 1260 1261 1262 1263
    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.

    """
1264 1265 1266 1267 1268 1269 1270
    # 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):
        return (os.path.exists(fn) and os.access(fn, mode)
                and not os.path.isdir(fn))

1271 1272 1273 1274 1275 1276 1277
    # 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
1278

1279 1280 1281 1282
    if path is None:
        path = os.environ.get("PATH", os.defpath)
    if not path:
        return None
1283
    path = path.split(os.pathsep)
1284 1285 1286 1287 1288 1289 1290 1291 1292 1293

    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".
1294 1295
        # If it does match, only test that one, otherwise we have to try
        # others.
1296 1297 1298 1299
        if any(cmd.lower().endswith(ext.lower()) for ext in pathext):
            files = [cmd]
        else:
            files = [cmd + ext for ext in pathext]
1300 1301 1302 1303 1304 1305 1306
    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:
1307 1308 1309
        normdir = os.path.normcase(dir)
        if not normdir in seen:
            seen.add(normdir)
1310 1311 1312 1313 1314
            for thefile in files:
                name = os.path.join(dir, thefile)
                if _access_check(name, mode):
                    return name
    return None