2000-02-04 23:28:42 +08:00
|
|
|
"""Utility functions for copying files and directory trees.
|
1997-04-29 22:45:19 +08:00
|
|
|
|
1999-08-19 04:03:17 +08:00
|
|
|
XXX The functions here don't copy the resource fork or other metadata on Mac.
|
1997-04-29 22:45:19 +08:00
|
|
|
|
|
|
|
"""
|
1990-10-14 03:23:40 +08:00
|
|
|
|
1992-04-01 02:55:40 +08:00
|
|
|
import os
|
1999-02-24 07:07:51 +08:00
|
|
|
import sys
|
1997-04-29 22:45:19 +08:00
|
|
|
import stat
|
2002-10-07 21:23:24 +08:00
|
|
|
import exceptions
|
2004-06-20 05:11:35 +08:00
|
|
|
from os.path import abspath
|
1990-10-14 03:23:40 +08:00
|
|
|
|
2001-02-16 06:15:14 +08:00
|
|
|
__all__ = ["copyfileobj","copyfile","copymode","copystat","copy","copy2",
|
2002-10-30 13:44:50 +08:00
|
|
|
"copytree","move","rmtree","Error"]
|
2002-10-07 21:23:24 +08:00
|
|
|
|
|
|
|
class Error(exceptions.EnvironmentError):
|
|
|
|
pass
|
1990-10-14 03:23:40 +08:00
|
|
|
|
2000-07-12 17:55:30 +08:00
|
|
|
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)
|
|
|
|
|
2001-01-15 09:36:40 +08:00
|
|
|
|
1990-10-14 03:23:40 +08:00
|
|
|
def copyfile(src, dst):
|
1997-04-29 22:45:19 +08:00
|
|
|
"""Copy data from src to dst"""
|
1997-04-29 22:06:46 +08:00
|
|
|
fsrc = None
|
|
|
|
fdst = None
|
2002-09-09 04:43:59 +08:00
|
|
|
# check for same pathname; all platforms
|
|
|
|
_src = os.path.normcase(os.path.abspath(src))
|
|
|
|
_dst = os.path.normcase(os.path.abspath(dst))
|
|
|
|
if _src == _dst:
|
|
|
|
return
|
1997-04-29 22:06:46 +08:00
|
|
|
try:
|
1998-03-27 05:13:24 +08:00
|
|
|
fsrc = open(src, 'rb')
|
|
|
|
fdst = open(dst, 'wb')
|
2000-07-12 17:55:30 +08:00
|
|
|
copyfileobj(fsrc, fdst)
|
1997-04-29 22:06:46 +08:00
|
|
|
finally:
|
1998-03-27 05:13:24 +08:00
|
|
|
if fdst:
|
|
|
|
fdst.close()
|
|
|
|
if fsrc:
|
|
|
|
fsrc.close()
|
1990-10-14 03:23:40 +08:00
|
|
|
|
|
|
|
def copymode(src, dst):
|
1997-04-29 22:45:19 +08:00
|
|
|
"""Copy mode bits from src to dst"""
|
2001-01-22 04:00:00 +08:00
|
|
|
if hasattr(os, 'chmod'):
|
|
|
|
st = os.stat(src)
|
2002-06-06 17:48:13 +08:00
|
|
|
mode = stat.S_IMODE(st.st_mode)
|
2001-01-22 04:00:00 +08:00
|
|
|
os.chmod(dst, mode)
|
1990-10-14 03:23:40 +08:00
|
|
|
|
|
|
|
def copystat(src, dst):
|
1997-04-29 22:45:19 +08:00
|
|
|
"""Copy all stat info (mode bits, atime and mtime) from src to dst"""
|
1997-04-29 22:06:46 +08:00
|
|
|
st = os.stat(src)
|
2002-06-06 17:48:13 +08:00
|
|
|
mode = stat.S_IMODE(st.st_mode)
|
2001-01-22 04:00:00 +08:00
|
|
|
if hasattr(os, 'utime'):
|
2002-06-06 17:48:13 +08:00
|
|
|
os.utime(dst, (st.st_atime, st.st_mtime))
|
2001-01-22 04:00:00 +08:00
|
|
|
if hasattr(os, 'chmod'):
|
|
|
|
os.chmod(dst, mode)
|
1997-04-29 22:45:19 +08:00
|
|
|
|
1990-10-14 03:23:40 +08:00
|
|
|
|
|
|
|
def copy(src, dst):
|
1997-04-29 22:45:19 +08:00
|
|
|
"""Copy data and mode bits ("cp src dst").
|
2001-01-15 09:36:40 +08:00
|
|
|
|
1997-04-29 22:45:19 +08:00
|
|
|
The destination may be a directory.
|
|
|
|
|
|
|
|
"""
|
1997-04-29 22:06:46 +08:00
|
|
|
if os.path.isdir(dst):
|
1998-03-27 05:13:24 +08:00
|
|
|
dst = os.path.join(dst, os.path.basename(src))
|
1997-04-29 22:06:46 +08:00
|
|
|
copyfile(src, dst)
|
|
|
|
copymode(src, dst)
|
1990-10-14 03:23:40 +08:00
|
|
|
|
|
|
|
def copy2(src, dst):
|
1997-04-29 22:45:19 +08:00
|
|
|
"""Copy data and all stat info ("cp -p src dst").
|
|
|
|
|
|
|
|
The destination may be a directory.
|
|
|
|
|
|
|
|
"""
|
1997-04-29 22:06:46 +08:00
|
|
|
if os.path.isdir(dst):
|
1998-03-27 05:13:24 +08:00
|
|
|
dst = os.path.join(dst, os.path.basename(src))
|
1997-04-29 22:06:46 +08:00
|
|
|
copyfile(src, dst)
|
|
|
|
copystat(src, dst)
|
1990-10-14 03:23:40 +08:00
|
|
|
|
1997-04-29 22:45:19 +08:00
|
|
|
|
2003-02-24 05:36:32 +08:00
|
|
|
def copytree(src, dst, symlinks=False):
|
1997-04-29 22:45:19 +08:00
|
|
|
"""Recursively copy a directory tree using copy2().
|
|
|
|
|
|
|
|
The destination directory must not already exist.
|
2003-02-24 05:36:32 +08:00
|
|
|
If exception(s) occur, an Error is raised with a list of reasons.
|
1997-04-29 22:45:19 +08:00
|
|
|
|
|
|
|
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
|
|
|
|
links are copied.
|
|
|
|
|
|
|
|
XXX Consider this example code rather than the ultimate tool.
|
|
|
|
|
|
|
|
"""
|
1997-04-29 22:06:46 +08:00
|
|
|
names = os.listdir(src)
|
1997-04-29 22:45:19 +08:00
|
|
|
os.mkdir(dst)
|
2002-10-07 21:23:24 +08:00
|
|
|
errors = []
|
1997-04-29 22:06:46 +08:00
|
|
|
for name in names:
|
1998-03-27 05:13:24 +08:00
|
|
|
srcname = os.path.join(src, name)
|
|
|
|
dstname = os.path.join(dst, name)
|
|
|
|
try:
|
|
|
|
if symlinks and os.path.islink(srcname):
|
|
|
|
linkto = os.readlink(srcname)
|
|
|
|
os.symlink(linkto, dstname)
|
|
|
|
elif os.path.isdir(srcname):
|
2000-04-07 22:34:50 +08:00
|
|
|
copytree(srcname, dstname, symlinks)
|
1998-03-27 05:13:24 +08:00
|
|
|
else:
|
|
|
|
copy2(srcname, dstname)
|
|
|
|
# XXX What about devices, sockets etc.?
|
|
|
|
except (IOError, os.error), why:
|
2002-10-07 21:23:24 +08:00
|
|
|
errors.append((srcname, dstname, why))
|
|
|
|
if errors:
|
|
|
|
raise Error, errors
|
1998-02-07 05:38:09 +08:00
|
|
|
|
2003-01-25 01:36:15 +08:00
|
|
|
def rmtree(path, ignore_errors=False, onerror=None):
|
1998-02-07 05:38:09 +08:00
|
|
|
"""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; otherwise, an
|
|
|
|
exception is raised.
|
|
|
|
"""
|
|
|
|
cmdtuples = []
|
2003-01-25 01:36:15 +08:00
|
|
|
arg = path
|
|
|
|
try:
|
|
|
|
_build_cmdtuple(path, cmdtuples)
|
|
|
|
for func, arg in cmdtuples:
|
2003-01-06 03:44:11 +08:00
|
|
|
func(arg)
|
2003-01-25 01:36:15 +08:00
|
|
|
except OSError:
|
|
|
|
exc = sys.exc_info()
|
|
|
|
if ignore_errors:
|
|
|
|
pass
|
|
|
|
elif onerror is not None:
|
|
|
|
onerror(func, arg, exc)
|
|
|
|
else:
|
|
|
|
raise exc[0], (exc[1][0], exc[1][1] + ' removing '+arg)
|
1998-02-07 05:38:09 +08:00
|
|
|
|
|
|
|
# Helper for rmtree()
|
|
|
|
def _build_cmdtuple(path, cmdtuples):
|
|
|
|
for f in os.listdir(path):
|
1998-03-27 05:13:24 +08:00
|
|
|
real_f = os.path.join(path,f)
|
|
|
|
if os.path.isdir(real_f) and not os.path.islink(real_f):
|
|
|
|
_build_cmdtuple(real_f, cmdtuples)
|
|
|
|
else:
|
1998-10-07 21:18:17 +08:00
|
|
|
cmdtuples.append((os.remove, real_f))
|
|
|
|
cmdtuples.append((os.rmdir, path))
|
2002-10-07 21:23:24 +08:00
|
|
|
|
|
|
|
|
|
|
|
def move(src, dst):
|
|
|
|
"""Recursively move a file or directory to another location.
|
|
|
|
|
|
|
|
If the destination is on our current filesystem, then simply use
|
|
|
|
rename. Otherwise, copy src to the dst and then remove src.
|
|
|
|
A lot more could be done here... A look at a mv.c shows a lot of
|
|
|
|
the issues this implementation glosses over.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
try:
|
|
|
|
os.rename(src, dst)
|
|
|
|
except OSError:
|
|
|
|
if os.path.isdir(src):
|
2004-06-20 05:11:35 +08:00
|
|
|
if destinsrc(src, dst):
|
|
|
|
raise Error, "Cannot move a directory '%s' into itself '%s'." % (src, dst)
|
2003-02-24 05:36:32 +08:00
|
|
|
copytree(src, dst, symlinks=True)
|
2002-10-07 21:23:24 +08:00
|
|
|
rmtree(src)
|
|
|
|
else:
|
|
|
|
copy2(src,dst)
|
|
|
|
os.unlink(src)
|
2004-06-20 05:11:35 +08:00
|
|
|
|
|
|
|
def destinsrc(src, dst):
|
|
|
|
return abspath(dst).startswith(abspath(src))
|