summaryrefslogtreecommitdiffstats
path: root/Lib
diff options
context:
space:
mode:
authorAntoine Pitrou <solipsis@pitrou.net>2009-05-01 20:55:35 (GMT)
committerAntoine Pitrou <solipsis@pitrou.net>2009-05-01 20:55:35 (GMT)
commit1fc0231a22f00afffff890e0cf0078dd7d0cf582 (patch)
treecfce54597979871a67d1948456b78a293b098ad3 /Lib
parent1d499265e08acac748b8158e90cf549194bac97c (diff)
downloadcpython-1fc0231a22f00afffff890e0cf0078dd7d0cf582.zip
cpython-1fc0231a22f00afffff890e0cf0078dd7d0cf582.tar.gz
cpython-1fc0231a22f00afffff890e0cf0078dd7d0cf582.tar.bz2
Issue #3002: `shutil.copyfile()` and `shutil.copytree()` now raise an
error when a named pipe is encountered, rather than blocking infinitely.
Diffstat (limited to 'Lib')
-rw-r--r--Lib/shutil.py21
-rw-r--r--Lib/test/test_shutil.py33
2 files changed, 50 insertions, 4 deletions
diff --git a/Lib/shutil.py b/Lib/shutil.py
index af96546..c46ec47 100644
--- a/Lib/shutil.py
+++ b/Lib/shutil.py
@@ -11,11 +11,15 @@ from os.path import abspath
import fnmatch
__all__ = ["copyfileobj","copyfile","copymode","copystat","copy","copy2",
- "copytree","move","rmtree","Error"]
+ "copytree","move","rmtree","Error", "SpecialFileError"]
class Error(EnvironmentError):
pass
+class SpecialFileError(EnvironmentError):
+ """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)"""
+
try:
WindowsError
except NameError:
@@ -48,6 +52,15 @@ def copyfile(src, dst):
fsrc = None
fdst = None
+ for fn in [src, dst]:
+ try:
+ st = os.stat(fn)
+ except OSError:
+ # File most likely does not exist
+ pass
+ # XXX What about other special files? (sockets, devices...)
+ if stat.S_ISFIFO(st.st_mode):
+ raise SpecialFileError("`%s` is a named pipe" % fn)
try:
fsrc = open(src, 'rb')
fdst = open(dst, 'wb')
@@ -157,14 +170,14 @@ def copytree(src, dst, symlinks=False, ignore=None):
elif os.path.isdir(srcname):
copytree(srcname, dstname, symlinks, ignore)
else:
+ # Will raise a SpecialFileError for unsupported file types
copy2(srcname, dstname)
- # XXX What about devices, sockets etc.?
- except (IOError, os.error), why:
- errors.append((srcname, dstname, str(why)))
# catch the Error from the recursive copytree so that we can
# continue with other files
except Error, err:
errors.extend(err.args[0])
+ except EnvironmentError, why:
+ errors.append((srcname, dstname, str(why)))
try:
copystat(src, dst)
except OSError, why:
diff --git a/Lib/test/test_shutil.py b/Lib/test/test_shutil.py
index 169290d..6226f9b 100644
--- a/Lib/test/test_shutil.py
+++ b/Lib/test/test_shutil.py
@@ -9,6 +9,7 @@ import os
import os.path
from test import test_support
from test.test_support import TESTFN
+TESTFN2 = TESTFN + "2"
class TestShutil(unittest.TestCase):
def test_rmtree_errors(self):
@@ -240,6 +241,38 @@ class TestShutil(unittest.TestCase):
finally:
shutil.rmtree(TESTFN, ignore_errors=True)
+ if hasattr(os, "mkfifo"):
+ # Issue #3002: copyfile and copytree block indefinitely on named pipes
+ def test_copyfile_named_pipe(self):
+ os.mkfifo(TESTFN)
+ try:
+ self.assertRaises(shutil.SpecialFileError,
+ shutil.copyfile, TESTFN, TESTFN2)
+ self.assertRaises(shutil.SpecialFileError,
+ shutil.copyfile, __file__, TESTFN)
+ finally:
+ os.remove(TESTFN)
+
+ def test_copytree_named_pipe(self):
+ os.mkdir(TESTFN)
+ try:
+ subdir = os.path.join(TESTFN, "subdir")
+ os.mkdir(subdir)
+ pipe = os.path.join(subdir, "mypipe")
+ os.mkfifo(pipe)
+ try:
+ shutil.copytree(TESTFN, TESTFN2)
+ except shutil.Error as e:
+ errors = e.args[0]
+ self.assertEqual(len(errors), 1)
+ src, dst, error_msg = errors[0]
+ self.assertEqual("`%s` is a named pipe" % pipe, error_msg)
+ else:
+ self.fail("shutil.Error should have been raised")
+ finally:
+ shutil.rmtree(TESTFN, ignore_errors=True)
+ shutil.rmtree(TESTFN2, ignore_errors=True)
+
class TestMove(unittest.TestCase):