blob: f21e6adf4abad037d1650d9c34d69801e7c64b1d [file] [log] [blame]
Barry Warsaw7fc2cca2003-01-24 17:34:13 +00001# Copyright (C) 2003 Python Software Foundation
2
3import unittest
Berker Peksag884afd92014-12-10 02:50:32 +02004import unittest.mock
Barry Warsaw7fc2cca2003-01-24 17:34:13 +00005import shutil
6import tempfile
Johannes Gijsbers8e6f2de2004-11-23 09:27:27 +00007import sys
Johannes Gijsbersef5ffc42004-10-31 12:05:31 +00008import stat
Brett Cannon1c3fa182004-06-19 21:11:35 +00009import os
10import os.path
Antoine Pitrouc041ab62012-01-02 19:18:02 +010011import errno
Nick Coghlan8ed3cf32011-03-16 14:05:35 -040012import functools
Jelle Zijlstraa12df7b2017-05-05 14:27:12 -070013import pathlib
Antoine Pitroubcf2b592012-02-08 23:28:36 +010014import subprocess
Serhiy Storchaka527ef072015-09-06 18:33:19 +030015from shutil import (make_archive,
Tarek Ziadé396fad72010-02-23 05:30:31 +000016 register_archive_format, unregister_archive_format,
Tarek Ziadé6ac91722010-04-28 17:51:36 +000017 get_archive_formats, Error, unpack_archive,
18 register_unpack_format, RegistryError,
Hynek Schlawack48653762012-10-07 12:49:58 +020019 unregister_unpack_format, get_unpack_formats,
20 SameFileError)
Tarek Ziadé396fad72010-02-23 05:30:31 +000021import tarfile
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +020022import zipfile
Tarek Ziadé396fad72010-02-23 05:30:31 +000023
24from test import support
Victor Stinnerd6debb22017-03-27 16:05:26 +020025from test.support import TESTFN, android_not_root
Serhiy Storchaka11213772014-08-06 18:50:19 +030026
Antoine Pitrou7fff0962009-05-01 21:09:44 +000027TESTFN2 = TESTFN + "2"
Barry Warsaw7fc2cca2003-01-24 17:34:13 +000028
Tarek Ziadé396fad72010-02-23 05:30:31 +000029try:
30 import grp
31 import pwd
32 UID_GID_SUPPORT = True
33except ImportError:
34 UID_GID_SUPPORT = False
35
Nick Coghlan8ed3cf32011-03-16 14:05:35 -040036def _fake_rename(*args, **kwargs):
37 # Pretend the destination path is on a different filesystem.
Antoine Pitrouc041ab62012-01-02 19:18:02 +010038 raise OSError(getattr(errno, 'EXDEV', 18), "Invalid cross-device link")
Nick Coghlan8ed3cf32011-03-16 14:05:35 -040039
40def mock_rename(func):
41 @functools.wraps(func)
42 def wrap(*args, **kwargs):
43 try:
44 builtin_rename = os.rename
45 os.rename = _fake_rename
46 return func(*args, **kwargs)
47 finally:
48 os.rename = builtin_rename
49 return wrap
50
Éric Araujoa7e33a12011-08-12 19:51:35 +020051def write_file(path, content, binary=False):
52 """Write *content* to a file located at *path*.
53
54 If *path* is a tuple instead of a string, os.path.join will be used to
55 make a path. If *binary* is true, the file will be opened in binary
56 mode.
57 """
58 if isinstance(path, tuple):
59 path = os.path.join(*path)
60 with open(path, 'wb' if binary else 'w') as fp:
61 fp.write(content)
62
63def read_file(path, binary=False):
64 """Return contents from a file located at *path*.
65
66 If *path* is a tuple instead of a string, os.path.join will be used to
67 make a path. If *binary* is true, the file will be opened in binary
68 mode.
69 """
70 if isinstance(path, tuple):
71 path = os.path.join(*path)
72 with open(path, 'rb' if binary else 'r') as fp:
73 return fp.read()
74
Serhiy Storchaka527ef072015-09-06 18:33:19 +030075def rlistdir(path):
76 res = []
77 for name in sorted(os.listdir(path)):
78 p = os.path.join(path, name)
79 if os.path.isdir(p) and not os.path.islink(p):
80 res.append(name + '/')
81 for n in rlistdir(p):
82 res.append(name + '/' + n)
83 else:
84 res.append(name)
85 return res
86
Éric Araujoa7e33a12011-08-12 19:51:35 +020087
Barry Warsaw7fc2cca2003-01-24 17:34:13 +000088class TestShutil(unittest.TestCase):
Tarek Ziadé396fad72010-02-23 05:30:31 +000089
90 def setUp(self):
91 super(TestShutil, self).setUp()
92 self.tempdirs = []
93
94 def tearDown(self):
95 super(TestShutil, self).tearDown()
96 while self.tempdirs:
97 d = self.tempdirs.pop()
98 shutil.rmtree(d, os.name in ('nt', 'cygwin'))
99
Tarek Ziadé396fad72010-02-23 05:30:31 +0000100
101 def mkdtemp(self):
102 """Create a temporary directory that will be cleaned up.
103
104 Returns the path of the directory.
105 """
106 d = tempfile.mkdtemp()
107 self.tempdirs.append(d)
108 return d
Tarek Ziadé5340db32010-04-19 22:30:51 +0000109
Hynek Schlawack3b527782012-06-25 13:27:31 +0200110 def test_rmtree_works_on_bytes(self):
111 tmp = self.mkdtemp()
112 victim = os.path.join(tmp, 'killme')
113 os.mkdir(victim)
114 write_file(os.path.join(victim, 'somefile'), 'foo')
115 victim = os.fsencode(victim)
116 self.assertIsInstance(victim, bytes)
Steve Dowere58571b2016-09-08 11:11:13 -0700117 shutil.rmtree(victim)
Hynek Schlawack3b527782012-06-25 13:27:31 +0200118
Hynek Schlawacka75cd1c2012-06-28 12:07:29 +0200119 @support.skip_unless_symlink
120 def test_rmtree_fails_on_symlink(self):
121 tmp = self.mkdtemp()
122 dir_ = os.path.join(tmp, 'dir')
123 os.mkdir(dir_)
124 link = os.path.join(tmp, 'link')
125 os.symlink(dir_, link)
126 self.assertRaises(OSError, shutil.rmtree, link)
127 self.assertTrue(os.path.exists(dir_))
Hynek Schlawackd16eacb2012-12-10 09:00:09 +0100128 self.assertTrue(os.path.lexists(link))
129 errors = []
130 def onerror(*args):
131 errors.append(args)
132 shutil.rmtree(link, onerror=onerror)
133 self.assertEqual(len(errors), 1)
134 self.assertIs(errors[0][0], os.path.islink)
135 self.assertEqual(errors[0][1], link)
136 self.assertIsInstance(errors[0][2][1], OSError)
Hynek Schlawacka75cd1c2012-06-28 12:07:29 +0200137
138 @support.skip_unless_symlink
139 def test_rmtree_works_on_symlinks(self):
140 tmp = self.mkdtemp()
141 dir1 = os.path.join(tmp, 'dir1')
142 dir2 = os.path.join(dir1, 'dir2')
143 dir3 = os.path.join(tmp, 'dir3')
144 for d in dir1, dir2, dir3:
145 os.mkdir(d)
146 file1 = os.path.join(tmp, 'file1')
147 write_file(file1, 'foo')
148 link1 = os.path.join(dir1, 'link1')
149 os.symlink(dir2, link1)
150 link2 = os.path.join(dir1, 'link2')
151 os.symlink(dir3, link2)
152 link3 = os.path.join(dir1, 'link3')
153 os.symlink(file1, link3)
154 # make sure symlinks are removed but not followed
155 shutil.rmtree(dir1)
156 self.assertFalse(os.path.exists(dir1))
157 self.assertTrue(os.path.exists(dir3))
158 self.assertTrue(os.path.exists(file1))
159
Barry Warsaw7fc2cca2003-01-24 17:34:13 +0000160 def test_rmtree_errors(self):
161 # filename is guaranteed not to exist
162 filename = tempfile.mktemp()
Hynek Schlawackb5501102012-12-10 09:11:25 +0100163 self.assertRaises(FileNotFoundError, shutil.rmtree, filename)
164 # test that ignore_errors option is honored
Hynek Schlawackd16eacb2012-12-10 09:00:09 +0100165 shutil.rmtree(filename, ignore_errors=True)
166
167 # existing file
168 tmpdir = self.mkdtemp()
Hynek Schlawackb5501102012-12-10 09:11:25 +0100169 write_file((tmpdir, "tstfile"), "")
Hynek Schlawackd16eacb2012-12-10 09:00:09 +0100170 filename = os.path.join(tmpdir, "tstfile")
Hynek Schlawackb5501102012-12-10 09:11:25 +0100171 with self.assertRaises(NotADirectoryError) as cm:
Hynek Schlawackd16eacb2012-12-10 09:00:09 +0100172 shutil.rmtree(filename)
Hynek Schlawack87f9b462012-12-10 16:29:57 +0100173 # The reason for this rather odd construct is that Windows sprinkles
174 # a \*.* at the end of file names. But only sometimes on some buildbots
175 possible_args = [filename, os.path.join(filename, '*.*')]
176 self.assertIn(cm.exception.filename, possible_args)
Hynek Schlawackd16eacb2012-12-10 09:00:09 +0100177 self.assertTrue(os.path.exists(filename))
Hynek Schlawackb5501102012-12-10 09:11:25 +0100178 # test that ignore_errors option is honored
Hynek Schlawackd16eacb2012-12-10 09:00:09 +0100179 shutil.rmtree(filename, ignore_errors=True)
180 self.assertTrue(os.path.exists(filename))
181 errors = []
182 def onerror(*args):
183 errors.append(args)
184 shutil.rmtree(filename, onerror=onerror)
185 self.assertEqual(len(errors), 2)
186 self.assertIs(errors[0][0], os.listdir)
187 self.assertEqual(errors[0][1], filename)
Hynek Schlawackb5501102012-12-10 09:11:25 +0100188 self.assertIsInstance(errors[0][2][1], NotADirectoryError)
Hynek Schlawack87f9b462012-12-10 16:29:57 +0100189 self.assertIn(errors[0][2][1].filename, possible_args)
Hynek Schlawackd16eacb2012-12-10 09:00:09 +0100190 self.assertIs(errors[1][0], os.rmdir)
191 self.assertEqual(errors[1][1], filename)
Hynek Schlawackb5501102012-12-10 09:11:25 +0100192 self.assertIsInstance(errors[1][2][1], NotADirectoryError)
Hynek Schlawack87f9b462012-12-10 16:29:57 +0100193 self.assertIn(errors[1][2][1].filename, possible_args)
Johannes Gijsbersef5ffc42004-10-31 12:05:31 +0000194
Johannes Gijsbersef5ffc42004-10-31 12:05:31 +0000195
Serhiy Storchaka43767632013-11-03 21:31:38 +0200196 @unittest.skipUnless(hasattr(os, 'chmod'), 'requires os.chmod()')
197 @unittest.skipIf(sys.platform[:6] == 'cygwin',
198 "This test can't be run on Cygwin (issue #1071513).")
199 @unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
200 "This test can't be run reliably as root (issue #1076467).")
201 def test_on_error(self):
202 self.errorState = 0
203 os.mkdir(TESTFN)
204 self.addCleanup(shutil.rmtree, TESTFN)
Antoine Pitrou2f8a75c2012-06-23 21:28:15 +0200205
Serhiy Storchaka43767632013-11-03 21:31:38 +0200206 self.child_file_path = os.path.join(TESTFN, 'a')
207 self.child_dir_path = os.path.join(TESTFN, 'b')
208 support.create_empty_file(self.child_file_path)
209 os.mkdir(self.child_dir_path)
210 old_dir_mode = os.stat(TESTFN).st_mode
211 old_child_file_mode = os.stat(self.child_file_path).st_mode
212 old_child_dir_mode = os.stat(self.child_dir_path).st_mode
213 # Make unwritable.
214 new_mode = stat.S_IREAD|stat.S_IEXEC
215 os.chmod(self.child_file_path, new_mode)
216 os.chmod(self.child_dir_path, new_mode)
217 os.chmod(TESTFN, new_mode)
Johannes Gijsbersef5ffc42004-10-31 12:05:31 +0000218
Serhiy Storchaka43767632013-11-03 21:31:38 +0200219 self.addCleanup(os.chmod, TESTFN, old_dir_mode)
220 self.addCleanup(os.chmod, self.child_file_path, old_child_file_mode)
221 self.addCleanup(os.chmod, self.child_dir_path, old_child_dir_mode)
Antoine Pitrou2f8a75c2012-06-23 21:28:15 +0200222
Serhiy Storchaka43767632013-11-03 21:31:38 +0200223 shutil.rmtree(TESTFN, onerror=self.check_args_to_onerror)
224 # Test whether onerror has actually been called.
225 self.assertEqual(self.errorState, 3,
226 "Expected call to onerror function did not happen.")
Johannes Gijsbersef5ffc42004-10-31 12:05:31 +0000227
Johannes Gijsbersef5ffc42004-10-31 12:05:31 +0000228 def check_args_to_onerror(self, func, arg, exc):
Benjamin Peterson25c95f12009-05-08 20:42:26 +0000229 # test_rmtree_errors deliberately runs rmtree
Hynek Schlawack67be92b2012-06-23 17:58:42 +0200230 # on a directory that is chmod 500, which will fail.
Benjamin Peterson25c95f12009-05-08 20:42:26 +0000231 # This function is run when shutil.rmtree fails.
232 # 99.9% of the time it initially fails to remove
233 # a file in the directory, so the first time through
234 # func is os.remove.
235 # However, some Linux machines running ZFS on
236 # FUSE experienced a failure earlier in the process
237 # at os.listdir. The first failure may legally
238 # be either.
Antoine Pitrou4f6e3f72012-06-23 22:05:11 +0200239 if self.errorState < 2:
Hynek Schlawack2100b422012-06-23 20:28:32 +0200240 if func is os.unlink:
Antoine Pitrou4f6e3f72012-06-23 22:05:11 +0200241 self.assertEqual(arg, self.child_file_path)
242 elif func is os.rmdir:
243 self.assertEqual(arg, self.child_dir_path)
Benjamin Peterson25c95f12009-05-08 20:42:26 +0000244 else:
Antoine Pitrou4f6e3f72012-06-23 22:05:11 +0200245 self.assertIs(func, os.listdir)
Hynek Schlawack67be92b2012-06-23 17:58:42 +0200246 self.assertIn(arg, [TESTFN, self.child_dir_path])
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000247 self.assertTrue(issubclass(exc[0], OSError))
Hynek Schlawack67be92b2012-06-23 17:58:42 +0200248 self.errorState += 1
Johannes Gijsbersef5ffc42004-10-31 12:05:31 +0000249 else:
250 self.assertEqual(func, os.rmdir)
251 self.assertEqual(arg, TESTFN)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000252 self.assertTrue(issubclass(exc[0], OSError))
Hynek Schlawack67be92b2012-06-23 17:58:42 +0200253 self.errorState = 3
254
255 def test_rmtree_does_not_choke_on_failing_lstat(self):
256 try:
257 orig_lstat = os.lstat
Hynek Schlawacka75cd1c2012-06-28 12:07:29 +0200258 def raiser(fn, *args, **kwargs):
Hynek Schlawack67be92b2012-06-23 17:58:42 +0200259 if fn != TESTFN:
260 raise OSError()
261 else:
262 return orig_lstat(fn)
263 os.lstat = raiser
264
265 os.mkdir(TESTFN)
266 write_file((TESTFN, 'foo'), 'foo')
267 shutil.rmtree(TESTFN)
268 finally:
269 os.lstat = orig_lstat
Barry Warsaw7fc2cca2003-01-24 17:34:13 +0000270
Antoine Pitrou78091e62011-12-29 18:54:15 +0100271 @unittest.skipUnless(hasattr(os, 'chmod'), 'requires os.chmod')
272 @support.skip_unless_symlink
273 def test_copymode_follow_symlinks(self):
274 tmp_dir = self.mkdtemp()
275 src = os.path.join(tmp_dir, 'foo')
276 dst = os.path.join(tmp_dir, 'bar')
277 src_link = os.path.join(tmp_dir, 'baz')
278 dst_link = os.path.join(tmp_dir, 'quux')
279 write_file(src, 'foo')
280 write_file(dst, 'foo')
281 os.symlink(src, src_link)
282 os.symlink(dst, dst_link)
283 os.chmod(src, stat.S_IRWXU|stat.S_IRWXG)
284 # file to file
285 os.chmod(dst, stat.S_IRWXO)
286 self.assertNotEqual(os.stat(src).st_mode, os.stat(dst).st_mode)
287 shutil.copymode(src, dst)
288 self.assertEqual(os.stat(src).st_mode, os.stat(dst).st_mode)
Antoine Pitrou3f48ac92014-01-01 02:50:45 +0100289 # On Windows, os.chmod does not follow symlinks (issue #15411)
290 if os.name != 'nt':
291 # follow src link
292 os.chmod(dst, stat.S_IRWXO)
293 shutil.copymode(src_link, dst)
294 self.assertEqual(os.stat(src).st_mode, os.stat(dst).st_mode)
295 # follow dst link
296 os.chmod(dst, stat.S_IRWXO)
297 shutil.copymode(src, dst_link)
298 self.assertEqual(os.stat(src).st_mode, os.stat(dst).st_mode)
299 # follow both links
300 os.chmod(dst, stat.S_IRWXO)
301 shutil.copymode(src_link, dst_link)
302 self.assertEqual(os.stat(src).st_mode, os.stat(dst).st_mode)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100303
304 @unittest.skipUnless(hasattr(os, 'lchmod'), 'requires os.lchmod')
305 @support.skip_unless_symlink
306 def test_copymode_symlink_to_symlink(self):
307 tmp_dir = self.mkdtemp()
308 src = os.path.join(tmp_dir, 'foo')
309 dst = os.path.join(tmp_dir, 'bar')
310 src_link = os.path.join(tmp_dir, 'baz')
311 dst_link = os.path.join(tmp_dir, 'quux')
312 write_file(src, 'foo')
313 write_file(dst, 'foo')
314 os.symlink(src, src_link)
315 os.symlink(dst, dst_link)
316 os.chmod(src, stat.S_IRWXU|stat.S_IRWXG)
317 os.chmod(dst, stat.S_IRWXU)
318 os.lchmod(src_link, stat.S_IRWXO|stat.S_IRWXG)
319 # link to link
320 os.lchmod(dst_link, stat.S_IRWXO)
Larry Hastingsb4038062012-07-15 10:57:38 -0700321 shutil.copymode(src_link, dst_link, follow_symlinks=False)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100322 self.assertEqual(os.lstat(src_link).st_mode,
323 os.lstat(dst_link).st_mode)
324 self.assertNotEqual(os.stat(src).st_mode, os.stat(dst).st_mode)
325 # src link - use chmod
326 os.lchmod(dst_link, stat.S_IRWXO)
Larry Hastingsb4038062012-07-15 10:57:38 -0700327 shutil.copymode(src_link, dst, follow_symlinks=False)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100328 self.assertEqual(os.stat(src).st_mode, os.stat(dst).st_mode)
329 # dst link - use chmod
330 os.lchmod(dst_link, stat.S_IRWXO)
Larry Hastingsb4038062012-07-15 10:57:38 -0700331 shutil.copymode(src, dst_link, follow_symlinks=False)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100332 self.assertEqual(os.stat(src).st_mode, os.stat(dst).st_mode)
333
334 @unittest.skipIf(hasattr(os, 'lchmod'), 'requires os.lchmod to be missing')
335 @support.skip_unless_symlink
336 def test_copymode_symlink_to_symlink_wo_lchmod(self):
337 tmp_dir = self.mkdtemp()
338 src = os.path.join(tmp_dir, 'foo')
339 dst = os.path.join(tmp_dir, 'bar')
340 src_link = os.path.join(tmp_dir, 'baz')
341 dst_link = os.path.join(tmp_dir, 'quux')
342 write_file(src, 'foo')
343 write_file(dst, 'foo')
344 os.symlink(src, src_link)
345 os.symlink(dst, dst_link)
Larry Hastingsb4038062012-07-15 10:57:38 -0700346 shutil.copymode(src_link, dst_link, follow_symlinks=False) # silent fail
Antoine Pitrou78091e62011-12-29 18:54:15 +0100347
348 @support.skip_unless_symlink
349 def test_copystat_symlinks(self):
350 tmp_dir = self.mkdtemp()
351 src = os.path.join(tmp_dir, 'foo')
352 dst = os.path.join(tmp_dir, 'bar')
353 src_link = os.path.join(tmp_dir, 'baz')
354 dst_link = os.path.join(tmp_dir, 'qux')
355 write_file(src, 'foo')
356 src_stat = os.stat(src)
357 os.utime(src, (src_stat.st_atime,
358 src_stat.st_mtime - 42.0)) # ensure different mtimes
359 write_file(dst, 'bar')
360 self.assertNotEqual(os.stat(src).st_mtime, os.stat(dst).st_mtime)
361 os.symlink(src, src_link)
362 os.symlink(dst, dst_link)
363 if hasattr(os, 'lchmod'):
364 os.lchmod(src_link, stat.S_IRWXO)
365 if hasattr(os, 'lchflags') and hasattr(stat, 'UF_NODUMP'):
366 os.lchflags(src_link, stat.UF_NODUMP)
367 src_link_stat = os.lstat(src_link)
368 # follow
369 if hasattr(os, 'lchmod'):
Larry Hastingsb4038062012-07-15 10:57:38 -0700370 shutil.copystat(src_link, dst_link, follow_symlinks=True)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100371 self.assertNotEqual(src_link_stat.st_mode, os.stat(dst).st_mode)
372 # don't follow
Larry Hastingsb4038062012-07-15 10:57:38 -0700373 shutil.copystat(src_link, dst_link, follow_symlinks=False)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100374 dst_link_stat = os.lstat(dst_link)
Larry Hastings9cf065c2012-06-22 16:30:09 -0700375 if os.utime in os.supports_follow_symlinks:
Antoine Pitrou78091e62011-12-29 18:54:15 +0100376 for attr in 'st_atime', 'st_mtime':
377 # The modification times may be truncated in the new file.
378 self.assertLessEqual(getattr(src_link_stat, attr),
379 getattr(dst_link_stat, attr) + 1)
380 if hasattr(os, 'lchmod'):
381 self.assertEqual(src_link_stat.st_mode, dst_link_stat.st_mode)
382 if hasattr(os, 'lchflags') and hasattr(src_link_stat, 'st_flags'):
383 self.assertEqual(src_link_stat.st_flags, dst_link_stat.st_flags)
384 # tell to follow but dst is not a link
Larry Hastingsb4038062012-07-15 10:57:38 -0700385 shutil.copystat(src_link, dst, follow_symlinks=False)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100386 self.assertTrue(abs(os.stat(src).st_mtime - os.stat(dst).st_mtime) <
387 00000.1)
388
Ned Deilybaf75712012-05-10 17:05:19 -0700389 @unittest.skipUnless(hasattr(os, 'chflags') and
390 hasattr(errno, 'EOPNOTSUPP') and
391 hasattr(errno, 'ENOTSUP'),
392 "requires os.chflags, EOPNOTSUPP & ENOTSUP")
393 def test_copystat_handles_harmless_chflags_errors(self):
394 tmpdir = self.mkdtemp()
395 file1 = os.path.join(tmpdir, 'file1')
396 file2 = os.path.join(tmpdir, 'file2')
397 write_file(file1, 'xxx')
398 write_file(file2, 'xxx')
399
400 def make_chflags_raiser(err):
401 ex = OSError()
402
Larry Hastings90867a52012-06-22 17:01:41 -0700403 def _chflags_raiser(path, flags, *, follow_symlinks=True):
Ned Deilybaf75712012-05-10 17:05:19 -0700404 ex.errno = err
405 raise ex
406 return _chflags_raiser
407 old_chflags = os.chflags
408 try:
409 for err in errno.EOPNOTSUPP, errno.ENOTSUP:
410 os.chflags = make_chflags_raiser(err)
411 shutil.copystat(file1, file2)
412 # assert others errors break it
413 os.chflags = make_chflags_raiser(errno.EOPNOTSUPP + errno.ENOTSUP)
414 self.assertRaises(OSError, shutil.copystat, file1, file2)
415 finally:
416 os.chflags = old_chflags
417
Antoine Pitrou424246f2012-05-12 19:02:01 +0200418 @support.skip_unless_xattr
419 def test_copyxattr(self):
420 tmp_dir = self.mkdtemp()
421 src = os.path.join(tmp_dir, 'foo')
422 write_file(src, 'foo')
423 dst = os.path.join(tmp_dir, 'bar')
424 write_file(dst, 'bar')
425
426 # no xattr == no problem
427 shutil._copyxattr(src, dst)
428 # common case
429 os.setxattr(src, 'user.foo', b'42')
430 os.setxattr(src, 'user.bar', b'43')
431 shutil._copyxattr(src, dst)
Gregory P. Smith1093bf22014-01-17 12:01:22 -0800432 self.assertEqual(sorted(os.listxattr(src)), sorted(os.listxattr(dst)))
Antoine Pitrou424246f2012-05-12 19:02:01 +0200433 self.assertEqual(
434 os.getxattr(src, 'user.foo'),
435 os.getxattr(dst, 'user.foo'))
436 # check errors don't affect other attrs
437 os.remove(dst)
438 write_file(dst, 'bar')
439 os_error = OSError(errno.EPERM, 'EPERM')
440
Larry Hastings9cf065c2012-06-22 16:30:09 -0700441 def _raise_on_user_foo(fname, attr, val, **kwargs):
Antoine Pitrou424246f2012-05-12 19:02:01 +0200442 if attr == 'user.foo':
443 raise os_error
444 else:
Larry Hastings9cf065c2012-06-22 16:30:09 -0700445 orig_setxattr(fname, attr, val, **kwargs)
Antoine Pitrou424246f2012-05-12 19:02:01 +0200446 try:
447 orig_setxattr = os.setxattr
448 os.setxattr = _raise_on_user_foo
449 shutil._copyxattr(src, dst)
Antoine Pitrou61597d32012-05-12 23:37:35 +0200450 self.assertIn('user.bar', os.listxattr(dst))
Antoine Pitrou424246f2012-05-12 19:02:01 +0200451 finally:
452 os.setxattr = orig_setxattr
Hynek Schlawack0beab052013-02-05 08:22:44 +0100453 # the source filesystem not supporting xattrs should be ok, too.
454 def _raise_on_src(fname, *, follow_symlinks=True):
455 if fname == src:
456 raise OSError(errno.ENOTSUP, 'Operation not supported')
457 return orig_listxattr(fname, follow_symlinks=follow_symlinks)
458 try:
459 orig_listxattr = os.listxattr
460 os.listxattr = _raise_on_src
461 shutil._copyxattr(src, dst)
462 finally:
463 os.listxattr = orig_listxattr
Antoine Pitrou424246f2012-05-12 19:02:01 +0200464
Larry Hastingsad5ae042012-07-14 17:55:11 -0700465 # test that shutil.copystat copies xattrs
466 src = os.path.join(tmp_dir, 'the_original')
467 write_file(src, src)
468 os.setxattr(src, 'user.the_value', b'fiddly')
469 dst = os.path.join(tmp_dir, 'the_copy')
470 write_file(dst, dst)
471 shutil.copystat(src, dst)
Hynek Schlawackc2d481f2012-07-16 17:11:10 +0200472 self.assertEqual(os.getxattr(dst, 'user.the_value'), b'fiddly')
Larry Hastingsad5ae042012-07-14 17:55:11 -0700473
Antoine Pitrou424246f2012-05-12 19:02:01 +0200474 @support.skip_unless_symlink
475 @support.skip_unless_xattr
476 @unittest.skipUnless(hasattr(os, 'geteuid') and os.geteuid() == 0,
477 'root privileges required')
478 def test_copyxattr_symlinks(self):
479 # On Linux, it's only possible to access non-user xattr for symlinks;
480 # which in turn require root privileges. This test should be expanded
481 # as soon as other platforms gain support for extended attributes.
482 tmp_dir = self.mkdtemp()
483 src = os.path.join(tmp_dir, 'foo')
484 src_link = os.path.join(tmp_dir, 'baz')
485 write_file(src, 'foo')
486 os.symlink(src, src_link)
487 os.setxattr(src, 'trusted.foo', b'42')
Larry Hastings9cf065c2012-06-22 16:30:09 -0700488 os.setxattr(src_link, 'trusted.foo', b'43', follow_symlinks=False)
Antoine Pitrou424246f2012-05-12 19:02:01 +0200489 dst = os.path.join(tmp_dir, 'bar')
490 dst_link = os.path.join(tmp_dir, 'qux')
491 write_file(dst, 'bar')
492 os.symlink(dst, dst_link)
Larry Hastingsb4038062012-07-15 10:57:38 -0700493 shutil._copyxattr(src_link, dst_link, follow_symlinks=False)
Larry Hastings9cf065c2012-06-22 16:30:09 -0700494 self.assertEqual(os.getxattr(dst_link, 'trusted.foo', follow_symlinks=False), b'43')
Antoine Pitrou424246f2012-05-12 19:02:01 +0200495 self.assertRaises(OSError, os.getxattr, dst, 'trusted.foo')
Larry Hastingsb4038062012-07-15 10:57:38 -0700496 shutil._copyxattr(src_link, dst, follow_symlinks=False)
Antoine Pitrou424246f2012-05-12 19:02:01 +0200497 self.assertEqual(os.getxattr(dst, 'trusted.foo'), b'43')
498
Antoine Pitrou78091e62011-12-29 18:54:15 +0100499 @support.skip_unless_symlink
500 def test_copy_symlinks(self):
501 tmp_dir = self.mkdtemp()
502 src = os.path.join(tmp_dir, 'foo')
503 dst = os.path.join(tmp_dir, 'bar')
504 src_link = os.path.join(tmp_dir, 'baz')
505 write_file(src, 'foo')
506 os.symlink(src, src_link)
507 if hasattr(os, 'lchmod'):
508 os.lchmod(src_link, stat.S_IRWXU | stat.S_IRWXO)
509 # don't follow
Larry Hastingsb4038062012-07-15 10:57:38 -0700510 shutil.copy(src_link, dst, follow_symlinks=True)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100511 self.assertFalse(os.path.islink(dst))
512 self.assertEqual(read_file(src), read_file(dst))
513 os.remove(dst)
514 # follow
Larry Hastingsb4038062012-07-15 10:57:38 -0700515 shutil.copy(src_link, dst, follow_symlinks=False)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100516 self.assertTrue(os.path.islink(dst))
517 self.assertEqual(os.readlink(dst), os.readlink(src_link))
518 if hasattr(os, 'lchmod'):
519 self.assertEqual(os.lstat(src_link).st_mode,
520 os.lstat(dst).st_mode)
521
522 @support.skip_unless_symlink
523 def test_copy2_symlinks(self):
524 tmp_dir = self.mkdtemp()
525 src = os.path.join(tmp_dir, 'foo')
526 dst = os.path.join(tmp_dir, 'bar')
527 src_link = os.path.join(tmp_dir, 'baz')
528 write_file(src, 'foo')
529 os.symlink(src, src_link)
530 if hasattr(os, 'lchmod'):
531 os.lchmod(src_link, stat.S_IRWXU | stat.S_IRWXO)
532 if hasattr(os, 'lchflags') and hasattr(stat, 'UF_NODUMP'):
533 os.lchflags(src_link, stat.UF_NODUMP)
534 src_stat = os.stat(src)
535 src_link_stat = os.lstat(src_link)
536 # follow
Larry Hastingsb4038062012-07-15 10:57:38 -0700537 shutil.copy2(src_link, dst, follow_symlinks=True)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100538 self.assertFalse(os.path.islink(dst))
539 self.assertEqual(read_file(src), read_file(dst))
540 os.remove(dst)
541 # don't follow
Larry Hastingsb4038062012-07-15 10:57:38 -0700542 shutil.copy2(src_link, dst, follow_symlinks=False)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100543 self.assertTrue(os.path.islink(dst))
544 self.assertEqual(os.readlink(dst), os.readlink(src_link))
545 dst_stat = os.lstat(dst)
Larry Hastings9cf065c2012-06-22 16:30:09 -0700546 if os.utime in os.supports_follow_symlinks:
Antoine Pitrou78091e62011-12-29 18:54:15 +0100547 for attr in 'st_atime', 'st_mtime':
548 # The modification times may be truncated in the new file.
549 self.assertLessEqual(getattr(src_link_stat, attr),
550 getattr(dst_stat, attr) + 1)
551 if hasattr(os, 'lchmod'):
552 self.assertEqual(src_link_stat.st_mode, dst_stat.st_mode)
553 self.assertNotEqual(src_stat.st_mode, dst_stat.st_mode)
554 if hasattr(os, 'lchflags') and hasattr(src_link_stat, 'st_flags'):
555 self.assertEqual(src_link_stat.st_flags, dst_stat.st_flags)
556
Antoine Pitrou424246f2012-05-12 19:02:01 +0200557 @support.skip_unless_xattr
558 def test_copy2_xattr(self):
559 tmp_dir = self.mkdtemp()
560 src = os.path.join(tmp_dir, 'foo')
561 dst = os.path.join(tmp_dir, 'bar')
562 write_file(src, 'foo')
563 os.setxattr(src, 'user.foo', b'42')
564 shutil.copy2(src, dst)
565 self.assertEqual(
566 os.getxattr(src, 'user.foo'),
567 os.getxattr(dst, 'user.foo'))
568 os.remove(dst)
569
Antoine Pitrou78091e62011-12-29 18:54:15 +0100570 @support.skip_unless_symlink
571 def test_copyfile_symlinks(self):
572 tmp_dir = self.mkdtemp()
573 src = os.path.join(tmp_dir, 'src')
574 dst = os.path.join(tmp_dir, 'dst')
575 dst_link = os.path.join(tmp_dir, 'dst_link')
576 link = os.path.join(tmp_dir, 'link')
577 write_file(src, 'foo')
578 os.symlink(src, link)
579 # don't follow
Larry Hastingsb4038062012-07-15 10:57:38 -0700580 shutil.copyfile(link, dst_link, follow_symlinks=False)
Antoine Pitrou78091e62011-12-29 18:54:15 +0100581 self.assertTrue(os.path.islink(dst_link))
582 self.assertEqual(os.readlink(link), os.readlink(dst_link))
583 # follow
584 shutil.copyfile(link, dst)
585 self.assertFalse(os.path.islink(dst))
586
Hynek Schlawack2100b422012-06-23 20:28:32 +0200587 def test_rmtree_uses_safe_fd_version_if_available(self):
Hynek Schlawackd0f6e0a2012-06-29 08:28:20 +0200588 _use_fd_functions = ({os.open, os.stat, os.unlink, os.rmdir} <=
589 os.supports_dir_fd and
590 os.listdir in os.supports_fd and
591 os.stat in os.supports_follow_symlinks)
592 if _use_fd_functions:
Hynek Schlawack2100b422012-06-23 20:28:32 +0200593 self.assertTrue(shutil._use_fd_functions)
Nick Coghlan5b0eca12012-06-24 16:43:06 +1000594 self.assertTrue(shutil.rmtree.avoids_symlink_attacks)
Hynek Schlawack2100b422012-06-23 20:28:32 +0200595 tmp_dir = self.mkdtemp()
596 d = os.path.join(tmp_dir, 'a')
597 os.mkdir(d)
598 try:
599 real_rmtree = shutil._rmtree_safe_fd
600 class Called(Exception): pass
601 def _raiser(*args, **kwargs):
602 raise Called
603 shutil._rmtree_safe_fd = _raiser
604 self.assertRaises(Called, shutil.rmtree, d)
605 finally:
606 shutil._rmtree_safe_fd = real_rmtree
607 else:
608 self.assertFalse(shutil._use_fd_functions)
Nick Coghlan5b0eca12012-06-24 16:43:06 +1000609 self.assertFalse(shutil.rmtree.avoids_symlink_attacks)
Hynek Schlawack2100b422012-06-23 20:28:32 +0200610
Johannes Gijsbersd60e92a2004-09-11 21:26:21 +0000611 def test_rmtree_dont_delete_file(self):
612 # When called on a file instead of a directory, don't delete it.
613 handle, path = tempfile.mkstemp()
Victor Stinnerbf816222011-06-30 23:25:47 +0200614 os.close(handle)
Hynek Schlawack67be92b2012-06-23 17:58:42 +0200615 self.assertRaises(NotADirectoryError, shutil.rmtree, path)
Johannes Gijsbersd60e92a2004-09-11 21:26:21 +0000616 os.remove(path)
617
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000618 def test_copytree_simple(self):
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000619 src_dir = tempfile.mkdtemp()
620 dst_dir = os.path.join(tempfile.mkdtemp(), 'destination')
Éric Araujoa7e33a12011-08-12 19:51:35 +0200621 self.addCleanup(shutil.rmtree, src_dir)
622 self.addCleanup(shutil.rmtree, os.path.dirname(dst_dir))
623 write_file((src_dir, 'test.txt'), '123')
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000624 os.mkdir(os.path.join(src_dir, 'test_dir'))
Éric Araujoa7e33a12011-08-12 19:51:35 +0200625 write_file((src_dir, 'test_dir', 'test.txt'), '456')
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000626
Éric Araujoa7e33a12011-08-12 19:51:35 +0200627 shutil.copytree(src_dir, dst_dir)
628 self.assertTrue(os.path.isfile(os.path.join(dst_dir, 'test.txt')))
629 self.assertTrue(os.path.isdir(os.path.join(dst_dir, 'test_dir')))
630 self.assertTrue(os.path.isfile(os.path.join(dst_dir, 'test_dir',
631 'test.txt')))
632 actual = read_file((dst_dir, 'test.txt'))
633 self.assertEqual(actual, '123')
634 actual = read_file((dst_dir, 'test_dir', 'test.txt'))
635 self.assertEqual(actual, '456')
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000636
Antoine Pitrou78091e62011-12-29 18:54:15 +0100637 @support.skip_unless_symlink
638 def test_copytree_symlinks(self):
639 tmp_dir = self.mkdtemp()
640 src_dir = os.path.join(tmp_dir, 'src')
641 dst_dir = os.path.join(tmp_dir, 'dst')
642 sub_dir = os.path.join(src_dir, 'sub')
643 os.mkdir(src_dir)
644 os.mkdir(sub_dir)
645 write_file((src_dir, 'file.txt'), 'foo')
646 src_link = os.path.join(sub_dir, 'link')
647 dst_link = os.path.join(dst_dir, 'sub/link')
648 os.symlink(os.path.join(src_dir, 'file.txt'),
649 src_link)
650 if hasattr(os, 'lchmod'):
651 os.lchmod(src_link, stat.S_IRWXU | stat.S_IRWXO)
652 if hasattr(os, 'lchflags') and hasattr(stat, 'UF_NODUMP'):
653 os.lchflags(src_link, stat.UF_NODUMP)
654 src_stat = os.lstat(src_link)
655 shutil.copytree(src_dir, dst_dir, symlinks=True)
656 self.assertTrue(os.path.islink(os.path.join(dst_dir, 'sub', 'link')))
657 self.assertEqual(os.readlink(os.path.join(dst_dir, 'sub', 'link')),
658 os.path.join(src_dir, 'file.txt'))
659 dst_stat = os.lstat(dst_link)
660 if hasattr(os, 'lchmod'):
661 self.assertEqual(dst_stat.st_mode, src_stat.st_mode)
662 if hasattr(os, 'lchflags'):
663 self.assertEqual(dst_stat.st_flags, src_stat.st_flags)
664
Georg Brandl2ee470f2008-07-16 12:55:28 +0000665 def test_copytree_with_exclude(self):
Georg Brandl2ee470f2008-07-16 12:55:28 +0000666 # creating data
667 join = os.path.join
668 exists = os.path.exists
669 src_dir = tempfile.mkdtemp()
Georg Brandl2ee470f2008-07-16 12:55:28 +0000670 try:
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000671 dst_dir = join(tempfile.mkdtemp(), 'destination')
Éric Araujoa7e33a12011-08-12 19:51:35 +0200672 write_file((src_dir, 'test.txt'), '123')
673 write_file((src_dir, 'test.tmp'), '123')
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000674 os.mkdir(join(src_dir, 'test_dir'))
Éric Araujoa7e33a12011-08-12 19:51:35 +0200675 write_file((src_dir, 'test_dir', 'test.txt'), '456')
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000676 os.mkdir(join(src_dir, 'test_dir2'))
Éric Araujoa7e33a12011-08-12 19:51:35 +0200677 write_file((src_dir, 'test_dir2', 'test.txt'), '456')
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000678 os.mkdir(join(src_dir, 'test_dir2', 'subdir'))
679 os.mkdir(join(src_dir, 'test_dir2', 'subdir2'))
Éric Araujoa7e33a12011-08-12 19:51:35 +0200680 write_file((src_dir, 'test_dir2', 'subdir', 'test.txt'), '456')
681 write_file((src_dir, 'test_dir2', 'subdir2', 'test.py'), '456')
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000682
683 # testing glob-like patterns
684 try:
685 patterns = shutil.ignore_patterns('*.tmp', 'test_dir2')
686 shutil.copytree(src_dir, dst_dir, ignore=patterns)
687 # checking the result: some elements should not be copied
688 self.assertTrue(exists(join(dst_dir, 'test.txt')))
Éric Araujoa7e33a12011-08-12 19:51:35 +0200689 self.assertFalse(exists(join(dst_dir, 'test.tmp')))
690 self.assertFalse(exists(join(dst_dir, 'test_dir2')))
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000691 finally:
Éric Araujoa7e33a12011-08-12 19:51:35 +0200692 shutil.rmtree(dst_dir)
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000693 try:
694 patterns = shutil.ignore_patterns('*.tmp', 'subdir*')
695 shutil.copytree(src_dir, dst_dir, ignore=patterns)
696 # checking the result: some elements should not be copied
Éric Araujoa7e33a12011-08-12 19:51:35 +0200697 self.assertFalse(exists(join(dst_dir, 'test.tmp')))
698 self.assertFalse(exists(join(dst_dir, 'test_dir2', 'subdir2')))
699 self.assertFalse(exists(join(dst_dir, 'test_dir2', 'subdir')))
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000700 finally:
Éric Araujoa7e33a12011-08-12 19:51:35 +0200701 shutil.rmtree(dst_dir)
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000702
703 # testing callable-style
704 try:
705 def _filter(src, names):
706 res = []
707 for name in names:
708 path = os.path.join(src, name)
709
710 if (os.path.isdir(path) and
711 path.split()[-1] == 'subdir'):
712 res.append(name)
713 elif os.path.splitext(path)[-1] in ('.py'):
714 res.append(name)
715 return res
716
717 shutil.copytree(src_dir, dst_dir, ignore=_filter)
718
719 # checking the result: some elements should not be copied
Éric Araujoa7e33a12011-08-12 19:51:35 +0200720 self.assertFalse(exists(join(dst_dir, 'test_dir2', 'subdir2',
721 'test.py')))
722 self.assertFalse(exists(join(dst_dir, 'test_dir2', 'subdir')))
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000723
724 finally:
Éric Araujoa7e33a12011-08-12 19:51:35 +0200725 shutil.rmtree(dst_dir)
Georg Brandl2ee470f2008-07-16 12:55:28 +0000726 finally:
Antoine Pitrou97c81ef2009-11-04 00:57:15 +0000727 shutil.rmtree(src_dir)
728 shutil.rmtree(os.path.dirname(dst_dir))
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000729
Antoine Pitrouac601602013-08-16 19:35:02 +0200730 def test_copytree_retains_permissions(self):
731 tmp_dir = tempfile.mkdtemp()
732 src_dir = os.path.join(tmp_dir, 'source')
733 os.mkdir(src_dir)
734 dst_dir = os.path.join(tmp_dir, 'destination')
735 self.addCleanup(shutil.rmtree, tmp_dir)
736
737 os.chmod(src_dir, 0o777)
738 write_file((src_dir, 'permissive.txt'), '123')
739 os.chmod(os.path.join(src_dir, 'permissive.txt'), 0o777)
740 write_file((src_dir, 'restrictive.txt'), '456')
741 os.chmod(os.path.join(src_dir, 'restrictive.txt'), 0o600)
742 restrictive_subdir = tempfile.mkdtemp(dir=src_dir)
743 os.chmod(restrictive_subdir, 0o600)
744
745 shutil.copytree(src_dir, dst_dir)
Brett Cannon9c7eb552013-08-23 14:38:11 -0400746 self.assertEqual(os.stat(src_dir).st_mode, os.stat(dst_dir).st_mode)
747 self.assertEqual(os.stat(os.path.join(src_dir, 'permissive.txt')).st_mode,
Antoine Pitrouac601602013-08-16 19:35:02 +0200748 os.stat(os.path.join(dst_dir, 'permissive.txt')).st_mode)
Brett Cannon9c7eb552013-08-23 14:38:11 -0400749 self.assertEqual(os.stat(os.path.join(src_dir, 'restrictive.txt')).st_mode,
Antoine Pitrouac601602013-08-16 19:35:02 +0200750 os.stat(os.path.join(dst_dir, 'restrictive.txt')).st_mode)
751 restrictive_subdir_dst = os.path.join(dst_dir,
752 os.path.split(restrictive_subdir)[1])
Brett Cannon9c7eb552013-08-23 14:38:11 -0400753 self.assertEqual(os.stat(restrictive_subdir).st_mode,
Antoine Pitrouac601602013-08-16 19:35:02 +0200754 os.stat(restrictive_subdir_dst).st_mode)
755
Berker Peksag884afd92014-12-10 02:50:32 +0200756 @unittest.mock.patch('os.chmod')
757 def test_copytree_winerror(self, mock_patch):
758 # When copying to VFAT, copystat() raises OSError. On Windows, the
759 # exception object has a meaningful 'winerror' attribute, but not
760 # on other operating systems. Do not assume 'winerror' is set.
761 src_dir = tempfile.mkdtemp()
762 dst_dir = os.path.join(tempfile.mkdtemp(), 'destination')
763 self.addCleanup(shutil.rmtree, src_dir)
764 self.addCleanup(shutil.rmtree, os.path.dirname(dst_dir))
765
766 mock_patch.side_effect = PermissionError('ka-boom')
767 with self.assertRaises(shutil.Error):
768 shutil.copytree(src_dir, dst_dir)
769
Zachary Ware9fe6d862013-12-08 00:20:35 -0600770 @unittest.skipIf(os.name == 'nt', 'temporarily disabled on Windows')
Hirokazu Yamamoto26681452010-12-05 02:04:16 +0000771 @unittest.skipUnless(hasattr(os, 'link'), 'requires os.link')
Xavier de Gaye3a4e9892016-12-13 10:00:01 +0100772 @unittest.skipIf(android_not_root, "hard links not allowed, non root user")
Tarek Ziadé51a6f722010-04-23 13:03:09 +0000773 def test_dont_copy_file_onto_link_to_itself(self):
774 # bug 851123.
775 os.mkdir(TESTFN)
776 src = os.path.join(TESTFN, 'cheese')
777 dst = os.path.join(TESTFN, 'shop')
778 try:
Hirokazu Yamamoto26681452010-12-05 02:04:16 +0000779 with open(src, 'w') as f:
780 f.write('cheddar')
781 os.link(src, dst)
Hynek Schlawack48653762012-10-07 12:49:58 +0200782 self.assertRaises(shutil.SameFileError, shutil.copyfile, src, dst)
Hirokazu Yamamoto26681452010-12-05 02:04:16 +0000783 with open(src, 'r') as f:
784 self.assertEqual(f.read(), 'cheddar')
785 os.remove(dst)
786 finally:
787 shutil.rmtree(TESTFN, ignore_errors=True)
Tarek Ziadé51a6f722010-04-23 13:03:09 +0000788
Brian Curtin3b4499c2010-12-28 14:31:47 +0000789 @support.skip_unless_symlink
Hirokazu Yamamoto26681452010-12-05 02:04:16 +0000790 def test_dont_copy_file_onto_symlink_to_itself(self):
791 # bug 851123.
792 os.mkdir(TESTFN)
793 src = os.path.join(TESTFN, 'cheese')
794 dst = os.path.join(TESTFN, 'shop')
795 try:
796 with open(src, 'w') as f:
797 f.write('cheddar')
Tarek Ziadé51a6f722010-04-23 13:03:09 +0000798 # Using `src` here would mean we end up with a symlink pointing
799 # to TESTFN/TESTFN/cheese, while it should point at
800 # TESTFN/cheese.
801 os.symlink('cheese', dst)
Hynek Schlawack48653762012-10-07 12:49:58 +0200802 self.assertRaises(shutil.SameFileError, shutil.copyfile, src, dst)
Antoine Pitrou92f60ed2010-10-14 22:11:44 +0000803 with open(src, 'r') as f:
804 self.assertEqual(f.read(), 'cheddar')
Tarek Ziadé51a6f722010-04-23 13:03:09 +0000805 os.remove(dst)
806 finally:
Hirokazu Yamamoto26681452010-12-05 02:04:16 +0000807 shutil.rmtree(TESTFN, ignore_errors=True)
Johannes Gijsbers68128712004-08-14 13:57:08 +0000808
Brian Curtin3b4499c2010-12-28 14:31:47 +0000809 @support.skip_unless_symlink
Brian Curtind40e6f72010-07-08 21:39:08 +0000810 def test_rmtree_on_symlink(self):
811 # bug 1669.
812 os.mkdir(TESTFN)
Tarek Ziadé51a6f722010-04-23 13:03:09 +0000813 try:
Brian Curtind40e6f72010-07-08 21:39:08 +0000814 src = os.path.join(TESTFN, 'cheese')
815 dst = os.path.join(TESTFN, 'shop')
816 os.mkdir(src)
817 os.symlink(src, dst)
818 self.assertRaises(OSError, shutil.rmtree, dst)
Hynek Schlawack67be92b2012-06-23 17:58:42 +0200819 shutil.rmtree(dst, ignore_errors=True)
Tarek Ziadé51a6f722010-04-23 13:03:09 +0000820 finally:
Brian Curtind40e6f72010-07-08 21:39:08 +0000821 shutil.rmtree(TESTFN, ignore_errors=True)
822
Serhiy Storchaka43767632013-11-03 21:31:38 +0200823 # Issue #3002: copyfile and copytree block indefinitely on named pipes
824 @unittest.skipUnless(hasattr(os, "mkfifo"), 'requires os.mkfifo()')
Xavier de Gaye3a4e9892016-12-13 10:00:01 +0100825 @unittest.skipIf(android_not_root, "mkfifo not allowed, non root user")
Serhiy Storchaka43767632013-11-03 21:31:38 +0200826 def test_copyfile_named_pipe(self):
827 os.mkfifo(TESTFN)
828 try:
829 self.assertRaises(shutil.SpecialFileError,
830 shutil.copyfile, TESTFN, TESTFN2)
831 self.assertRaises(shutil.SpecialFileError,
832 shutil.copyfile, __file__, TESTFN)
833 finally:
834 os.remove(TESTFN)
Antoine Pitrou7fff0962009-05-01 21:09:44 +0000835
Xavier de Gaye3a4e9892016-12-13 10:00:01 +0100836 @unittest.skipIf(android_not_root, "mkfifo not allowed, non root user")
Serhiy Storchaka43767632013-11-03 21:31:38 +0200837 @unittest.skipUnless(hasattr(os, "mkfifo"), 'requires os.mkfifo()')
838 @support.skip_unless_symlink
839 def test_copytree_named_pipe(self):
840 os.mkdir(TESTFN)
841 try:
842 subdir = os.path.join(TESTFN, "subdir")
843 os.mkdir(subdir)
844 pipe = os.path.join(subdir, "mypipe")
845 os.mkfifo(pipe)
Antoine Pitrou7fff0962009-05-01 21:09:44 +0000846 try:
Serhiy Storchaka43767632013-11-03 21:31:38 +0200847 shutil.copytree(TESTFN, TESTFN2)
848 except shutil.Error as e:
849 errors = e.args[0]
850 self.assertEqual(len(errors), 1)
851 src, dst, error_msg = errors[0]
852 self.assertEqual("`%s` is a named pipe" % pipe, error_msg)
853 else:
854 self.fail("shutil.Error should have been raised")
855 finally:
856 shutil.rmtree(TESTFN, ignore_errors=True)
857 shutil.rmtree(TESTFN2, ignore_errors=True)
Antoine Pitrou7fff0962009-05-01 21:09:44 +0000858
Tarek Ziadé5340db32010-04-19 22:30:51 +0000859 def test_copytree_special_func(self):
860
861 src_dir = self.mkdtemp()
862 dst_dir = os.path.join(self.mkdtemp(), 'destination')
Éric Araujoa7e33a12011-08-12 19:51:35 +0200863 write_file((src_dir, 'test.txt'), '123')
Tarek Ziadé5340db32010-04-19 22:30:51 +0000864 os.mkdir(os.path.join(src_dir, 'test_dir'))
Éric Araujoa7e33a12011-08-12 19:51:35 +0200865 write_file((src_dir, 'test_dir', 'test.txt'), '456')
Tarek Ziadé5340db32010-04-19 22:30:51 +0000866
867 copied = []
868 def _copy(src, dst):
869 copied.append((src, dst))
870
871 shutil.copytree(src_dir, dst_dir, copy_function=_copy)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000872 self.assertEqual(len(copied), 2)
Tarek Ziadé5340db32010-04-19 22:30:51 +0000873
Brian Curtin3b4499c2010-12-28 14:31:47 +0000874 @support.skip_unless_symlink
Tarek Ziadéfb437512010-04-20 08:57:33 +0000875 def test_copytree_dangling_symlinks(self):
876
877 # a dangling symlink raises an error at the end
878 src_dir = self.mkdtemp()
879 dst_dir = os.path.join(self.mkdtemp(), 'destination')
880 os.symlink('IDONTEXIST', os.path.join(src_dir, 'test.txt'))
881 os.mkdir(os.path.join(src_dir, 'test_dir'))
Éric Araujoa7e33a12011-08-12 19:51:35 +0200882 write_file((src_dir, 'test_dir', 'test.txt'), '456')
Tarek Ziadéfb437512010-04-20 08:57:33 +0000883 self.assertRaises(Error, shutil.copytree, src_dir, dst_dir)
884
885 # a dangling symlink is ignored with the proper flag
886 dst_dir = os.path.join(self.mkdtemp(), 'destination2')
887 shutil.copytree(src_dir, dst_dir, ignore_dangling_symlinks=True)
888 self.assertNotIn('test.txt', os.listdir(dst_dir))
889
890 # a dangling symlink is copied if symlinks=True
891 dst_dir = os.path.join(self.mkdtemp(), 'destination3')
892 shutil.copytree(src_dir, dst_dir, symlinks=True)
893 self.assertIn('test.txt', os.listdir(dst_dir))
894
Berker Peksag5a294d82015-07-25 14:53:48 +0300895 @support.skip_unless_symlink
896 def test_copytree_symlink_dir(self):
897 src_dir = self.mkdtemp()
898 dst_dir = os.path.join(self.mkdtemp(), 'destination')
899 os.mkdir(os.path.join(src_dir, 'real_dir'))
900 with open(os.path.join(src_dir, 'real_dir', 'test.txt'), 'w'):
901 pass
902 os.symlink(os.path.join(src_dir, 'real_dir'),
903 os.path.join(src_dir, 'link_to_dir'),
904 target_is_directory=True)
905
906 shutil.copytree(src_dir, dst_dir, symlinks=False)
907 self.assertFalse(os.path.islink(os.path.join(dst_dir, 'link_to_dir')))
908 self.assertIn('test.txt', os.listdir(os.path.join(dst_dir, 'link_to_dir')))
909
910 dst_dir = os.path.join(self.mkdtemp(), 'destination2')
911 shutil.copytree(src_dir, dst_dir, symlinks=True)
912 self.assertTrue(os.path.islink(os.path.join(dst_dir, 'link_to_dir')))
913 self.assertIn('test.txt', os.listdir(os.path.join(dst_dir, 'link_to_dir')))
914
Nick Coghlan8ed3cf32011-03-16 14:05:35 -0400915 def _copy_file(self, method):
916 fname = 'test.txt'
917 tmpdir = self.mkdtemp()
Éric Araujoa7e33a12011-08-12 19:51:35 +0200918 write_file((tmpdir, fname), 'xxx')
Nick Coghlan8ed3cf32011-03-16 14:05:35 -0400919 file1 = os.path.join(tmpdir, fname)
920 tmpdir2 = self.mkdtemp()
921 method(file1, tmpdir2)
922 file2 = os.path.join(tmpdir2, fname)
923 return (file1, file2)
924
925 @unittest.skipUnless(hasattr(os, 'chmod'), 'requires os.chmod')
926 def test_copy(self):
927 # Ensure that the copied file exists and has the same mode bits.
928 file1, file2 = self._copy_file(shutil.copy)
929 self.assertTrue(os.path.exists(file2))
930 self.assertEqual(os.stat(file1).st_mode, os.stat(file2).st_mode)
931
932 @unittest.skipUnless(hasattr(os, 'chmod'), 'requires os.chmod')
Senthil Kumaran0c2dba52011-07-03 18:21:38 -0700933 @unittest.skipUnless(hasattr(os, 'utime'), 'requires os.utime')
Nick Coghlan8ed3cf32011-03-16 14:05:35 -0400934 def test_copy2(self):
935 # Ensure that the copied file exists and has the same mode and
936 # modification time bits.
937 file1, file2 = self._copy_file(shutil.copy2)
938 self.assertTrue(os.path.exists(file2))
939 file1_stat = os.stat(file1)
940 file2_stat = os.stat(file2)
941 self.assertEqual(file1_stat.st_mode, file2_stat.st_mode)
942 for attr in 'st_atime', 'st_mtime':
943 # The modification times may be truncated in the new file.
944 self.assertLessEqual(getattr(file1_stat, attr),
945 getattr(file2_stat, attr) + 1)
946 if hasattr(os, 'chflags') and hasattr(file1_stat, 'st_flags'):
947 self.assertEqual(getattr(file1_stat, 'st_flags'),
948 getattr(file2_stat, 'st_flags'))
949
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +0200950 @support.requires_zlib
Tarek Ziadé396fad72010-02-23 05:30:31 +0000951 def test_make_tarball(self):
952 # creating something to tar
Serhiy Storchaka527ef072015-09-06 18:33:19 +0300953 root_dir, base_dir = self._create_files('')
Tarek Ziadé396fad72010-02-23 05:30:31 +0000954
955 tmpdir2 = self.mkdtemp()
Nick Coghlan8ed3cf32011-03-16 14:05:35 -0400956 # force shutil to create the directory
957 os.rmdir(tmpdir2)
Serhiy Storchakaeba8fee2015-09-07 19:58:23 +0300958 # working with relative paths
959 work_dir = os.path.dirname(tmpdir2)
960 rel_base_name = os.path.join(os.path.basename(tmpdir2), 'archive')
Tarek Ziadé396fad72010-02-23 05:30:31 +0000961
Serhiy Storchakaeba8fee2015-09-07 19:58:23 +0300962 with support.change_cwd(work_dir):
Serhiy Storchaka5558d4f2015-09-08 09:59:02 +0300963 base_name = os.path.abspath(rel_base_name)
Serhiy Storchakaeba8fee2015-09-07 19:58:23 +0300964 tarball = make_archive(rel_base_name, 'gztar', root_dir, '.')
Tarek Ziadé396fad72010-02-23 05:30:31 +0000965
966 # check if the compressed tarball was created
Serhiy Storchakaa091a822015-09-07 13:55:25 +0300967 self.assertEqual(tarball, base_name + '.tar.gz')
Serhiy Storchaka527ef072015-09-06 18:33:19 +0300968 self.assertTrue(os.path.isfile(tarball))
969 self.assertTrue(tarfile.is_tarfile(tarball))
970 with tarfile.open(tarball, 'r:gz') as tf:
971 self.assertCountEqual(tf.getnames(),
972 ['.', './sub', './sub2',
973 './file1', './file2', './sub/file3'])
Tarek Ziadé396fad72010-02-23 05:30:31 +0000974
975 # trying an uncompressed one
Serhiy Storchakaeba8fee2015-09-07 19:58:23 +0300976 with support.change_cwd(work_dir):
977 tarball = make_archive(rel_base_name, 'tar', root_dir, '.')
Serhiy Storchakaa091a822015-09-07 13:55:25 +0300978 self.assertEqual(tarball, base_name + '.tar')
Serhiy Storchaka527ef072015-09-06 18:33:19 +0300979 self.assertTrue(os.path.isfile(tarball))
980 self.assertTrue(tarfile.is_tarfile(tarball))
981 with tarfile.open(tarball, 'r') as tf:
982 self.assertCountEqual(tf.getnames(),
983 ['.', './sub', './sub2',
984 './file1', './file2', './sub/file3'])
Tarek Ziadé396fad72010-02-23 05:30:31 +0000985
986 def _tarinfo(self, path):
Serhiy Storchaka527ef072015-09-06 18:33:19 +0300987 with tarfile.open(path) as tar:
Tarek Ziadé396fad72010-02-23 05:30:31 +0000988 names = tar.getnames()
989 names.sort()
990 return tuple(names)
Tarek Ziadé396fad72010-02-23 05:30:31 +0000991
Serhiy Storchaka527ef072015-09-06 18:33:19 +0300992 def _create_files(self, base_dir='dist'):
Tarek Ziadé396fad72010-02-23 05:30:31 +0000993 # creating something to tar
Serhiy Storchaka527ef072015-09-06 18:33:19 +0300994 root_dir = self.mkdtemp()
995 dist = os.path.join(root_dir, base_dir)
996 os.makedirs(dist, exist_ok=True)
Éric Araujoa7e33a12011-08-12 19:51:35 +0200997 write_file((dist, 'file1'), 'xxx')
998 write_file((dist, 'file2'), 'xxx')
Tarek Ziadé396fad72010-02-23 05:30:31 +0000999 os.mkdir(os.path.join(dist, 'sub'))
Éric Araujoa7e33a12011-08-12 19:51:35 +02001000 write_file((dist, 'sub', 'file3'), 'xxx')
Tarek Ziadé396fad72010-02-23 05:30:31 +00001001 os.mkdir(os.path.join(dist, 'sub2'))
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001002 if base_dir:
1003 write_file((root_dir, 'outer'), 'xxx')
1004 return root_dir, base_dir
Tarek Ziadé396fad72010-02-23 05:30:31 +00001005
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001006 @support.requires_zlib
Serhiy Storchakab42de2f2015-11-21 14:09:26 +02001007 @unittest.skipUnless(shutil.which('tar'),
Tarek Ziadé396fad72010-02-23 05:30:31 +00001008 'Need the tar command to run')
1009 def test_tarfile_vs_tar(self):
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001010 root_dir, base_dir = self._create_files()
1011 base_name = os.path.join(self.mkdtemp(), 'archive')
Serhiy Storchakaa091a822015-09-07 13:55:25 +03001012 tarball = make_archive(base_name, 'gztar', root_dir, base_dir)
Tarek Ziadé396fad72010-02-23 05:30:31 +00001013
1014 # check if the compressed tarball was created
Serhiy Storchakaa091a822015-09-07 13:55:25 +03001015 self.assertEqual(tarball, base_name + '.tar.gz')
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001016 self.assertTrue(os.path.isfile(tarball))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001017
1018 # now create another tarball using `tar`
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001019 tarball2 = os.path.join(root_dir, 'archive2.tar')
1020 tar_cmd = ['tar', '-cf', 'archive2.tar', base_dir]
Serhiy Storchakab42de2f2015-11-21 14:09:26 +02001021 subprocess.check_call(tar_cmd, cwd=root_dir,
1022 stdout=subprocess.DEVNULL)
Tarek Ziadé396fad72010-02-23 05:30:31 +00001023
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001024 self.assertTrue(os.path.isfile(tarball2))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001025 # let's compare both tarballs
Ezio Melottib3aedd42010-11-20 19:04:17 +00001026 self.assertEqual(self._tarinfo(tarball), self._tarinfo(tarball2))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001027
1028 # trying an uncompressed one
Serhiy Storchakaa091a822015-09-07 13:55:25 +03001029 tarball = make_archive(base_name, 'tar', root_dir, base_dir)
1030 self.assertEqual(tarball, base_name + '.tar')
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001031 self.assertTrue(os.path.isfile(tarball))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001032
1033 # now for a dry_run
Serhiy Storchakaa091a822015-09-07 13:55:25 +03001034 tarball = make_archive(base_name, 'tar', root_dir, base_dir,
1035 dry_run=True)
1036 self.assertEqual(tarball, base_name + '.tar')
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001037 self.assertTrue(os.path.isfile(tarball))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001038
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001039 @support.requires_zlib
Tarek Ziadé396fad72010-02-23 05:30:31 +00001040 def test_make_zipfile(self):
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001041 # creating something to zip
1042 root_dir, base_dir = self._create_files()
Serhiy Storchakaeba8fee2015-09-07 19:58:23 +03001043
1044 tmpdir2 = self.mkdtemp()
1045 # force shutil to create the directory
1046 os.rmdir(tmpdir2)
1047 # working with relative paths
1048 work_dir = os.path.dirname(tmpdir2)
1049 rel_base_name = os.path.join(os.path.basename(tmpdir2), 'archive')
Serhiy Storchakaeba8fee2015-09-07 19:58:23 +03001050
1051 with support.change_cwd(work_dir):
Serhiy Storchaka5558d4f2015-09-08 09:59:02 +03001052 base_name = os.path.abspath(rel_base_name)
Serhiy Storchaka666de772016-10-23 15:55:09 +03001053 res = make_archive(rel_base_name, 'zip', root_dir)
1054
1055 self.assertEqual(res, base_name + '.zip')
1056 self.assertTrue(os.path.isfile(res))
1057 self.assertTrue(zipfile.is_zipfile(res))
1058 with zipfile.ZipFile(res) as zf:
1059 self.assertCountEqual(zf.namelist(),
1060 ['dist/', 'dist/sub/', 'dist/sub2/',
1061 'dist/file1', 'dist/file2', 'dist/sub/file3',
1062 'outer'])
1063
1064 with support.change_cwd(work_dir):
1065 base_name = os.path.abspath(rel_base_name)
Serhiy Storchaka2504cec2015-09-08 05:47:23 +03001066 res = make_archive(rel_base_name, 'zip', root_dir, base_dir)
Tarek Ziadé396fad72010-02-23 05:30:31 +00001067
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001068 self.assertEqual(res, base_name + '.zip')
1069 self.assertTrue(os.path.isfile(res))
1070 self.assertTrue(zipfile.is_zipfile(res))
1071 with zipfile.ZipFile(res) as zf:
1072 self.assertCountEqual(zf.namelist(),
Serhiy Storchaka2504cec2015-09-08 05:47:23 +03001073 ['dist/', 'dist/sub/', 'dist/sub2/',
1074 'dist/file1', 'dist/file2', 'dist/sub/file3'])
Tarek Ziadé396fad72010-02-23 05:30:31 +00001075
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001076 @support.requires_zlib
Serhiy Storchakab42de2f2015-11-21 14:09:26 +02001077 @unittest.skipUnless(shutil.which('zip'),
Serhiy Storchaka2504cec2015-09-08 05:47:23 +03001078 'Need the zip command to run')
1079 def test_zipfile_vs_zip(self):
1080 root_dir, base_dir = self._create_files()
1081 base_name = os.path.join(self.mkdtemp(), 'archive')
1082 archive = make_archive(base_name, 'zip', root_dir, base_dir)
1083
1084 # check if ZIP file was created
1085 self.assertEqual(archive, base_name + '.zip')
1086 self.assertTrue(os.path.isfile(archive))
1087
1088 # now create another ZIP file using `zip`
1089 archive2 = os.path.join(root_dir, 'archive2.zip')
1090 zip_cmd = ['zip', '-q', '-r', 'archive2.zip', base_dir]
Serhiy Storchakab42de2f2015-11-21 14:09:26 +02001091 subprocess.check_call(zip_cmd, cwd=root_dir,
1092 stdout=subprocess.DEVNULL)
Serhiy Storchaka2504cec2015-09-08 05:47:23 +03001093
1094 self.assertTrue(os.path.isfile(archive2))
1095 # let's compare both ZIP files
1096 with zipfile.ZipFile(archive) as zf:
1097 names = zf.namelist()
1098 with zipfile.ZipFile(archive2) as zf:
1099 names2 = zf.namelist()
1100 self.assertEqual(sorted(names), sorted(names2))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001101
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001102 @support.requires_zlib
Serhiy Storchaka8bc792a2015-11-22 14:49:58 +02001103 @unittest.skipUnless(shutil.which('unzip'),
1104 'Need the unzip command to run')
1105 def test_unzip_zipfile(self):
1106 root_dir, base_dir = self._create_files()
1107 base_name = os.path.join(self.mkdtemp(), 'archive')
1108 archive = make_archive(base_name, 'zip', root_dir, base_dir)
1109
1110 # check if ZIP file was created
1111 self.assertEqual(archive, base_name + '.zip')
1112 self.assertTrue(os.path.isfile(archive))
1113
1114 # now check the ZIP file using `unzip -t`
1115 zip_cmd = ['unzip', '-t', archive]
1116 with support.change_cwd(root_dir):
1117 try:
1118 subprocess.check_output(zip_cmd, stderr=subprocess.STDOUT)
1119 except subprocess.CalledProcessError as exc:
1120 details = exc.output.decode(errors="replace")
1121 msg = "{}\n\n**Unzip Output**\n{}"
1122 self.fail(msg.format(exc, details))
1123
Tarek Ziadé396fad72010-02-23 05:30:31 +00001124 def test_make_archive(self):
1125 tmpdir = self.mkdtemp()
1126 base_name = os.path.join(tmpdir, 'archive')
1127 self.assertRaises(ValueError, make_archive, base_name, 'xxx')
1128
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001129 @support.requires_zlib
Tarek Ziadé396fad72010-02-23 05:30:31 +00001130 def test_make_archive_owner_group(self):
1131 # testing make_archive with owner and group, with various combinations
1132 # this works even if there's not gid/uid support
1133 if UID_GID_SUPPORT:
1134 group = grp.getgrgid(0)[0]
1135 owner = pwd.getpwuid(0)[0]
1136 else:
1137 group = owner = 'root'
1138
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001139 root_dir, base_dir = self._create_files()
1140 base_name = os.path.join(self.mkdtemp(), 'archive')
Tarek Ziadé396fad72010-02-23 05:30:31 +00001141 res = make_archive(base_name, 'zip', root_dir, base_dir, owner=owner,
1142 group=group)
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001143 self.assertTrue(os.path.isfile(res))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001144
1145 res = make_archive(base_name, 'zip', root_dir, base_dir)
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001146 self.assertTrue(os.path.isfile(res))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001147
1148 res = make_archive(base_name, 'tar', root_dir, base_dir,
1149 owner=owner, group=group)
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001150 self.assertTrue(os.path.isfile(res))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001151
1152 res = make_archive(base_name, 'tar', root_dir, base_dir,
1153 owner='kjhkjhkjg', group='oihohoh')
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001154 self.assertTrue(os.path.isfile(res))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001155
Tarek Ziadé6ac91722010-04-28 17:51:36 +00001156
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001157 @support.requires_zlib
Tarek Ziadé396fad72010-02-23 05:30:31 +00001158 @unittest.skipUnless(UID_GID_SUPPORT, "Requires grp and pwd support")
1159 def test_tarfile_root_owner(self):
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001160 root_dir, base_dir = self._create_files()
1161 base_name = os.path.join(self.mkdtemp(), 'archive')
Tarek Ziadé396fad72010-02-23 05:30:31 +00001162 group = grp.getgrgid(0)[0]
1163 owner = pwd.getpwuid(0)[0]
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001164 with support.change_cwd(root_dir):
1165 archive_name = make_archive(base_name, 'gztar', root_dir, 'dist',
1166 owner=owner, group=group)
Tarek Ziadé396fad72010-02-23 05:30:31 +00001167
1168 # check if the compressed tarball was created
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001169 self.assertTrue(os.path.isfile(archive_name))
Tarek Ziadé396fad72010-02-23 05:30:31 +00001170
1171 # now checks the rights
1172 archive = tarfile.open(archive_name)
1173 try:
1174 for member in archive.getmembers():
Ezio Melottib3aedd42010-11-20 19:04:17 +00001175 self.assertEqual(member.uid, 0)
1176 self.assertEqual(member.gid, 0)
Tarek Ziadé396fad72010-02-23 05:30:31 +00001177 finally:
1178 archive.close()
1179
1180 def test_make_archive_cwd(self):
1181 current_dir = os.getcwd()
1182 def _breaks(*args, **kw):
1183 raise RuntimeError()
1184
1185 register_archive_format('xxx', _breaks, [], 'xxx file')
1186 try:
1187 try:
1188 make_archive('xxx', 'xxx', root_dir=self.mkdtemp())
1189 except Exception:
1190 pass
Ezio Melottib3aedd42010-11-20 19:04:17 +00001191 self.assertEqual(os.getcwd(), current_dir)
Tarek Ziadé396fad72010-02-23 05:30:31 +00001192 finally:
1193 unregister_archive_format('xxx')
1194
Serhiy Storchaka9a4fc192014-11-28 00:48:46 +02001195 def test_make_tarfile_in_curdir(self):
1196 # Issue #21280
1197 root_dir = self.mkdtemp()
1198 with support.change_cwd(root_dir):
1199 self.assertEqual(make_archive('test', 'tar'), 'test.tar')
1200 self.assertTrue(os.path.isfile('test.tar'))
1201
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001202 @support.requires_zlib
Serhiy Storchaka9a4fc192014-11-28 00:48:46 +02001203 def test_make_zipfile_in_curdir(self):
1204 # Issue #21280
1205 root_dir = self.mkdtemp()
1206 with support.change_cwd(root_dir):
1207 self.assertEqual(make_archive('test', 'zip'), 'test.zip')
1208 self.assertTrue(os.path.isfile('test.zip'))
1209
Tarek Ziadé396fad72010-02-23 05:30:31 +00001210 def test_register_archive_format(self):
1211
1212 self.assertRaises(TypeError, register_archive_format, 'xxx', 1)
1213 self.assertRaises(TypeError, register_archive_format, 'xxx', lambda: x,
1214 1)
1215 self.assertRaises(TypeError, register_archive_format, 'xxx', lambda: x,
1216 [(1, 2), (1, 2, 3)])
1217
1218 register_archive_format('xxx', lambda: x, [(1, 2)], 'xxx file')
1219 formats = [name for name, params in get_archive_formats()]
1220 self.assertIn('xxx', formats)
1221
1222 unregister_archive_format('xxx')
1223 formats = [name for name, params in get_archive_formats()]
1224 self.assertNotIn('xxx', formats)
1225
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001226 def check_unpack_archive(self, format):
Jelle Zijlstraa12df7b2017-05-05 14:27:12 -07001227 self.check_unpack_archive_with_converter(format, lambda path: path)
1228 self.check_unpack_archive_with_converter(format, pathlib.Path)
1229
1230 class MyPath:
1231 def __init__(self, path):
1232 self.path = path
1233 def __fspath__(self):
1234 return self.path
1235
1236 self.check_unpack_archive_with_converter(format, MyPath)
1237
1238 def check_unpack_archive_with_converter(self, format, converter):
Serhiy Storchaka527ef072015-09-06 18:33:19 +03001239 root_dir, base_dir = self._create_files()
Serhiy Storchaka2504cec2015-09-08 05:47:23 +03001240 expected = rlistdir(root_dir)
1241 expected.remove('outer')
Tarek Ziadé6ac91722010-04-28 17:51:36 +00001242
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001243 base_name = os.path.join(self.mkdtemp(), 'archive')
1244 filename = make_archive(base_name, format, root_dir, base_dir)
Tarek Ziadé6ac91722010-04-28 17:51:36 +00001245
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001246 # let's try to unpack it now
1247 tmpdir2 = self.mkdtemp()
Jelle Zijlstraa12df7b2017-05-05 14:27:12 -07001248 unpack_archive(converter(filename), converter(tmpdir2))
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001249 self.assertEqual(rlistdir(tmpdir2), expected)
1250
1251 # and again, this time with the format specified
1252 tmpdir3 = self.mkdtemp()
Jelle Zijlstraa12df7b2017-05-05 14:27:12 -07001253 unpack_archive(converter(filename), converter(tmpdir3), format=format)
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001254 self.assertEqual(rlistdir(tmpdir3), expected)
1255
Jelle Zijlstraa12df7b2017-05-05 14:27:12 -07001256 self.assertRaises(shutil.ReadError, unpack_archive, converter(TESTFN))
1257 self.assertRaises(ValueError, unpack_archive, converter(TESTFN), format='xxx')
Nick Coghlanabf202d2011-03-16 13:52:20 -04001258
Serhiy Storchaka20cdffd2016-12-16 18:58:33 +02001259 def test_unpack_archive_tar(self):
1260 self.check_unpack_archive('tar')
1261
1262 @support.requires_zlib
1263 def test_unpack_archive_gztar(self):
1264 self.check_unpack_archive('gztar')
1265
1266 @support.requires_bz2
1267 def test_unpack_archive_bztar(self):
1268 self.check_unpack_archive('bztar')
1269
1270 @support.requires_lzma
1271 def test_unpack_archive_xztar(self):
1272 self.check_unpack_archive('xztar')
1273
1274 @support.requires_zlib
1275 def test_unpack_archive_zip(self):
1276 self.check_unpack_archive('zip')
1277
Martin Pantereb995702016-07-28 01:11:04 +00001278 def test_unpack_registry(self):
Tarek Ziadé6ac91722010-04-28 17:51:36 +00001279
1280 formats = get_unpack_formats()
1281
1282 def _boo(filename, extract_dir, extra):
Ezio Melottib3aedd42010-11-20 19:04:17 +00001283 self.assertEqual(extra, 1)
1284 self.assertEqual(filename, 'stuff.boo')
1285 self.assertEqual(extract_dir, 'xx')
Tarek Ziadé6ac91722010-04-28 17:51:36 +00001286
1287 register_unpack_format('Boo', ['.boo', '.b2'], _boo, [('extra', 1)])
1288 unpack_archive('stuff.boo', 'xx')
1289
1290 # trying to register a .boo unpacker again
1291 self.assertRaises(RegistryError, register_unpack_format, 'Boo2',
1292 ['.boo'], _boo)
1293
1294 # should work now
1295 unregister_unpack_format('Boo')
1296 register_unpack_format('Boo2', ['.boo'], _boo)
1297 self.assertIn(('Boo2', ['.boo'], ''), get_unpack_formats())
1298 self.assertNotIn(('Boo', ['.boo'], ''), get_unpack_formats())
1299
1300 # let's leave a clean state
1301 unregister_unpack_format('Boo2')
Ezio Melottib3aedd42010-11-20 19:04:17 +00001302 self.assertEqual(get_unpack_formats(), formats)
Tarek Ziadé6ac91722010-04-28 17:51:36 +00001303
Giampaolo Rodola'210e7ca2011-07-01 13:55:36 +02001304 @unittest.skipUnless(hasattr(shutil, 'disk_usage'),
1305 "disk_usage not available on this platform")
1306 def test_disk_usage(self):
1307 usage = shutil.disk_usage(os.getcwd())
Éric Araujo2ee61882011-07-02 16:45:45 +02001308 self.assertGreater(usage.total, 0)
1309 self.assertGreater(usage.used, 0)
1310 self.assertGreaterEqual(usage.free, 0)
1311 self.assertGreaterEqual(usage.total, usage.used)
1312 self.assertGreater(usage.total, usage.free)
Giampaolo Rodola'210e7ca2011-07-01 13:55:36 +02001313
Sandro Tosid902a142011-08-22 23:28:27 +02001314 @unittest.skipUnless(UID_GID_SUPPORT, "Requires grp and pwd support")
1315 @unittest.skipUnless(hasattr(os, 'chown'), 'requires os.chown')
1316 def test_chown(self):
1317
1318 # cleaned-up automatically by TestShutil.tearDown method
1319 dirname = self.mkdtemp()
1320 filename = tempfile.mktemp(dir=dirname)
1321 write_file(filename, 'testing chown function')
1322
1323 with self.assertRaises(ValueError):
1324 shutil.chown(filename)
1325
1326 with self.assertRaises(LookupError):
Raymond Hettinger15f44ab2016-08-30 10:47:49 -07001327 shutil.chown(filename, user='non-existing username')
Sandro Tosid902a142011-08-22 23:28:27 +02001328
1329 with self.assertRaises(LookupError):
Raymond Hettinger15f44ab2016-08-30 10:47:49 -07001330 shutil.chown(filename, group='non-existing groupname')
Sandro Tosid902a142011-08-22 23:28:27 +02001331
1332 with self.assertRaises(TypeError):
1333 shutil.chown(filename, b'spam')
1334
1335 with self.assertRaises(TypeError):
1336 shutil.chown(filename, 3.14)
1337
1338 uid = os.getuid()
1339 gid = os.getgid()
1340
1341 def check_chown(path, uid=None, gid=None):
1342 s = os.stat(filename)
1343 if uid is not None:
1344 self.assertEqual(uid, s.st_uid)
1345 if gid is not None:
1346 self.assertEqual(gid, s.st_gid)
1347
1348 shutil.chown(filename, uid, gid)
1349 check_chown(filename, uid, gid)
1350 shutil.chown(filename, uid)
1351 check_chown(filename, uid)
1352 shutil.chown(filename, user=uid)
1353 check_chown(filename, uid)
1354 shutil.chown(filename, group=gid)
Sandro Tosi91f948a2011-08-22 23:55:39 +02001355 check_chown(filename, gid=gid)
Sandro Tosid902a142011-08-22 23:28:27 +02001356
1357 shutil.chown(dirname, uid, gid)
1358 check_chown(dirname, uid, gid)
1359 shutil.chown(dirname, uid)
1360 check_chown(dirname, uid)
1361 shutil.chown(dirname, user=uid)
1362 check_chown(dirname, uid)
1363 shutil.chown(dirname, group=gid)
Sandro Tosi91f948a2011-08-22 23:55:39 +02001364 check_chown(dirname, gid=gid)
Sandro Tosid902a142011-08-22 23:28:27 +02001365
1366 user = pwd.getpwuid(uid)[0]
1367 group = grp.getgrgid(gid)[0]
1368 shutil.chown(filename, user, group)
1369 check_chown(filename, uid, gid)
1370 shutil.chown(dirname, user, group)
1371 check_chown(dirname, uid, gid)
1372
Brian Curtin0d0a1de2012-06-18 18:41:07 -05001373 def test_copy_return_value(self):
1374 # copy and copy2 both return their destination path.
1375 for fn in (shutil.copy, shutil.copy2):
1376 src_dir = self.mkdtemp()
1377 dst_dir = self.mkdtemp()
1378 src = os.path.join(src_dir, 'foo')
1379 write_file(src, 'foo')
1380 rv = fn(src, dst_dir)
1381 self.assertEqual(rv, os.path.join(dst_dir, 'foo'))
1382 rv = fn(src, os.path.join(dst_dir, 'bar'))
1383 self.assertEqual(rv, os.path.join(dst_dir, 'bar'))
1384
1385 def test_copyfile_return_value(self):
1386 # copytree returns its destination path.
1387 src_dir = self.mkdtemp()
1388 dst_dir = self.mkdtemp()
1389 dst_file = os.path.join(dst_dir, 'bar')
1390 src_file = os.path.join(src_dir, 'foo')
1391 write_file(src_file, 'foo')
1392 rv = shutil.copyfile(src_file, dst_file)
1393 self.assertTrue(os.path.exists(rv))
1394 self.assertEqual(read_file(src_file), read_file(dst_file))
1395
Hynek Schlawack48653762012-10-07 12:49:58 +02001396 def test_copyfile_same_file(self):
1397 # copyfile() should raise SameFileError if the source and destination
1398 # are the same.
1399 src_dir = self.mkdtemp()
1400 src_file = os.path.join(src_dir, 'foo')
1401 write_file(src_file, 'foo')
1402 self.assertRaises(SameFileError, shutil.copyfile, src_file, src_file)
Hynek Schlawack27ddb572012-10-28 13:59:27 +01001403 # But Error should work too, to stay backward compatible.
1404 self.assertRaises(Error, shutil.copyfile, src_file, src_file)
Hynek Schlawack48653762012-10-07 12:49:58 +02001405
Brian Curtin0d0a1de2012-06-18 18:41:07 -05001406 def test_copytree_return_value(self):
1407 # copytree returns its destination path.
1408 src_dir = self.mkdtemp()
1409 dst_dir = src_dir + "dest"
Larry Hastings5b2f9c02012-06-25 23:50:01 -07001410 self.addCleanup(shutil.rmtree, dst_dir, True)
Brian Curtin0d0a1de2012-06-18 18:41:07 -05001411 src = os.path.join(src_dir, 'foo')
1412 write_file(src, 'foo')
1413 rv = shutil.copytree(src_dir, dst_dir)
1414 self.assertEqual(['foo'], os.listdir(rv))
1415
Christian Heimes9bd667a2008-01-20 15:14:11 +00001416
Brian Curtinc57a3452012-06-22 16:00:30 -05001417class TestWhich(unittest.TestCase):
1418
1419 def setUp(self):
Serhiy Storchaka014791f2013-01-21 15:00:27 +02001420 self.temp_dir = tempfile.mkdtemp(prefix="Tmp")
Larry Hastings5b2f9c02012-06-25 23:50:01 -07001421 self.addCleanup(shutil.rmtree, self.temp_dir, True)
Brian Curtinc57a3452012-06-22 16:00:30 -05001422 # Give the temp_file an ".exe" suffix for all.
1423 # It's needed on Windows and not harmful on other platforms.
1424 self.temp_file = tempfile.NamedTemporaryFile(dir=self.temp_dir,
Serhiy Storchaka014791f2013-01-21 15:00:27 +02001425 prefix="Tmp",
1426 suffix=".Exe")
Brian Curtinc57a3452012-06-22 16:00:30 -05001427 os.chmod(self.temp_file.name, stat.S_IXUSR)
1428 self.addCleanup(self.temp_file.close)
1429 self.dir, self.file = os.path.split(self.temp_file.name)
1430
1431 def test_basic(self):
1432 # Given an EXE in a directory, it should be returned.
1433 rv = shutil.which(self.file, path=self.dir)
1434 self.assertEqual(rv, self.temp_file.name)
1435
Serhiy Storchaka8bea2002013-01-23 10:44:21 +02001436 def test_absolute_cmd(self):
Brian Curtinc57a3452012-06-22 16:00:30 -05001437 # When given the fully qualified path to an executable that exists,
1438 # it should be returned.
1439 rv = shutil.which(self.temp_file.name, path=self.temp_dir)
Serhiy Storchaka8bea2002013-01-23 10:44:21 +02001440 self.assertEqual(rv, self.temp_file.name)
1441
1442 def test_relative_cmd(self):
1443 # When given the relative path with a directory part to an executable
1444 # that exists, it should be returned.
1445 base_dir, tail_dir = os.path.split(self.dir)
1446 relpath = os.path.join(tail_dir, self.file)
Nick Coghlan55175962013-07-28 22:11:50 +10001447 with support.change_cwd(path=base_dir):
Serhiy Storchaka8bea2002013-01-23 10:44:21 +02001448 rv = shutil.which(relpath, path=self.temp_dir)
1449 self.assertEqual(rv, relpath)
1450 # But it shouldn't be searched in PATH directories (issue #16957).
Nick Coghlan55175962013-07-28 22:11:50 +10001451 with support.change_cwd(path=self.dir):
Serhiy Storchaka8bea2002013-01-23 10:44:21 +02001452 rv = shutil.which(relpath, path=base_dir)
1453 self.assertIsNone(rv)
1454
1455 def test_cwd(self):
1456 # Issue #16957
1457 base_dir = os.path.dirname(self.dir)
Nick Coghlan55175962013-07-28 22:11:50 +10001458 with support.change_cwd(path=self.dir):
Serhiy Storchaka8bea2002013-01-23 10:44:21 +02001459 rv = shutil.which(self.file, path=base_dir)
1460 if sys.platform == "win32":
1461 # Windows: current directory implicitly on PATH
1462 self.assertEqual(rv, os.path.join(os.curdir, self.file))
1463 else:
1464 # Other platforms: shouldn't match in the current directory.
1465 self.assertIsNone(rv)
Brian Curtinc57a3452012-06-22 16:00:30 -05001466
Serhiy Storchaka12516e22013-05-28 15:50:15 +03001467 @unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
1468 'non-root user required')
Brian Curtinc57a3452012-06-22 16:00:30 -05001469 def test_non_matching_mode(self):
1470 # Set the file read-only and ask for writeable files.
1471 os.chmod(self.temp_file.name, stat.S_IREAD)
Serhiy Storchaka12516e22013-05-28 15:50:15 +03001472 if os.access(self.temp_file.name, os.W_OK):
1473 self.skipTest("can't set the file read-only")
Brian Curtinc57a3452012-06-22 16:00:30 -05001474 rv = shutil.which(self.file, path=self.dir, mode=os.W_OK)
1475 self.assertIsNone(rv)
1476
Serhiy Storchaka8bea2002013-01-23 10:44:21 +02001477 def test_relative_path(self):
Antoine Pitrou07c24d12012-06-22 23:33:05 +02001478 base_dir, tail_dir = os.path.split(self.dir)
Nick Coghlan55175962013-07-28 22:11:50 +10001479 with support.change_cwd(path=base_dir):
Antoine Pitrou07c24d12012-06-22 23:33:05 +02001480 rv = shutil.which(self.file, path=tail_dir)
1481 self.assertEqual(rv, os.path.join(tail_dir, self.file))
Antoine Pitrou07c24d12012-06-22 23:33:05 +02001482
Brian Curtinc57a3452012-06-22 16:00:30 -05001483 def test_nonexistent_file(self):
1484 # Return None when no matching executable file is found on the path.
1485 rv = shutil.which("foo.exe", path=self.dir)
1486 self.assertIsNone(rv)
1487
1488 @unittest.skipUnless(sys.platform == "win32",
1489 "pathext check is Windows-only")
1490 def test_pathext_checking(self):
1491 # Ask for the file without the ".exe" extension, then ensure that
1492 # it gets found properly with the extension.
Serhiy Storchakad70127a2013-01-24 20:03:49 +02001493 rv = shutil.which(self.file[:-4], path=self.dir)
Serhiy Storchaka80c88f42013-01-22 10:31:36 +02001494 self.assertEqual(rv, self.temp_file.name[:-4] + ".EXE")
Brian Curtinc57a3452012-06-22 16:00:30 -05001495
Barry Warsaw618738b2013-04-16 11:05:03 -04001496 def test_environ_path(self):
1497 with support.EnvironmentVarGuard() as env:
Victor Stinner1d006a22013-12-16 23:39:40 +01001498 env['PATH'] = self.dir
Barry Warsaw618738b2013-04-16 11:05:03 -04001499 rv = shutil.which(self.file)
1500 self.assertEqual(rv, self.temp_file.name)
1501
1502 def test_empty_path(self):
1503 base_dir = os.path.dirname(self.dir)
Nick Coghlan55175962013-07-28 22:11:50 +10001504 with support.change_cwd(path=self.dir), \
Barry Warsaw618738b2013-04-16 11:05:03 -04001505 support.EnvironmentVarGuard() as env:
Victor Stinner1d006a22013-12-16 23:39:40 +01001506 env['PATH'] = self.dir
Barry Warsaw618738b2013-04-16 11:05:03 -04001507 rv = shutil.which(self.file, path='')
1508 self.assertIsNone(rv)
1509
1510 def test_empty_path_no_PATH(self):
1511 with support.EnvironmentVarGuard() as env:
1512 env.pop('PATH', None)
1513 rv = shutil.which(self.file)
1514 self.assertIsNone(rv)
1515
Brian Curtinc57a3452012-06-22 16:00:30 -05001516
Christian Heimesada8c3b2008-03-18 18:26:33 +00001517class TestMove(unittest.TestCase):
1518
1519 def setUp(self):
1520 filename = "foo"
1521 self.src_dir = tempfile.mkdtemp()
1522 self.dst_dir = tempfile.mkdtemp()
1523 self.src_file = os.path.join(self.src_dir, filename)
1524 self.dst_file = os.path.join(self.dst_dir, filename)
Christian Heimesada8c3b2008-03-18 18:26:33 +00001525 with open(self.src_file, "wb") as f:
1526 f.write(b"spam")
1527
1528 def tearDown(self):
Nick Coghlan8ed3cf32011-03-16 14:05:35 -04001529 for d in (self.src_dir, self.dst_dir):
Christian Heimesada8c3b2008-03-18 18:26:33 +00001530 try:
1531 if d:
1532 shutil.rmtree(d)
1533 except:
1534 pass
1535
1536 def _check_move_file(self, src, dst, real_dst):
Antoine Pitrou92f60ed2010-10-14 22:11:44 +00001537 with open(src, "rb") as f:
1538 contents = f.read()
Christian Heimesada8c3b2008-03-18 18:26:33 +00001539 shutil.move(src, dst)
Antoine Pitrou92f60ed2010-10-14 22:11:44 +00001540 with open(real_dst, "rb") as f:
1541 self.assertEqual(contents, f.read())
Christian Heimesada8c3b2008-03-18 18:26:33 +00001542 self.assertFalse(os.path.exists(src))
1543
1544 def _check_move_dir(self, src, dst, real_dst):
1545 contents = sorted(os.listdir(src))
1546 shutil.move(src, dst)
1547 self.assertEqual(contents, sorted(os.listdir(real_dst)))
1548 self.assertFalse(os.path.exists(src))
1549
1550 def test_move_file(self):
1551 # Move a file to another location on the same filesystem.
1552 self._check_move_file(self.src_file, self.dst_file, self.dst_file)
1553
1554 def test_move_file_to_dir(self):
1555 # Move a file inside an existing dir on the same filesystem.
1556 self._check_move_file(self.src_file, self.dst_dir, self.dst_file)
1557
Nick Coghlan8ed3cf32011-03-16 14:05:35 -04001558 @mock_rename
Christian Heimesada8c3b2008-03-18 18:26:33 +00001559 def test_move_file_other_fs(self):
1560 # Move a file to an existing dir on another filesystem.
Nick Coghlan8ed3cf32011-03-16 14:05:35 -04001561 self.test_move_file()
Christian Heimesada8c3b2008-03-18 18:26:33 +00001562
Nick Coghlan8ed3cf32011-03-16 14:05:35 -04001563 @mock_rename
Christian Heimesada8c3b2008-03-18 18:26:33 +00001564 def test_move_file_to_dir_other_fs(self):
1565 # Move a file to another location on another filesystem.
Nick Coghlan8ed3cf32011-03-16 14:05:35 -04001566 self.test_move_file_to_dir()
Christian Heimesada8c3b2008-03-18 18:26:33 +00001567
1568 def test_move_dir(self):
1569 # Move a dir to another location on the same filesystem.
1570 dst_dir = tempfile.mktemp()
1571 try:
1572 self._check_move_dir(self.src_dir, dst_dir, dst_dir)
1573 finally:
1574 try:
1575 shutil.rmtree(dst_dir)
1576 except:
1577 pass
1578
Nick Coghlan8ed3cf32011-03-16 14:05:35 -04001579 @mock_rename
Christian Heimesada8c3b2008-03-18 18:26:33 +00001580 def test_move_dir_other_fs(self):
1581 # Move a dir to another location on another filesystem.
Nick Coghlan8ed3cf32011-03-16 14:05:35 -04001582 self.test_move_dir()
Christian Heimesada8c3b2008-03-18 18:26:33 +00001583
1584 def test_move_dir_to_dir(self):
1585 # Move a dir inside an existing dir on the same filesystem.
1586 self._check_move_dir(self.src_dir, self.dst_dir,
1587 os.path.join(self.dst_dir, os.path.basename(self.src_dir)))
1588
Nick Coghlan8ed3cf32011-03-16 14:05:35 -04001589 @mock_rename
Christian Heimesada8c3b2008-03-18 18:26:33 +00001590 def test_move_dir_to_dir_other_fs(self):
1591 # Move a dir inside an existing dir on another filesystem.
Nick Coghlan8ed3cf32011-03-16 14:05:35 -04001592 self.test_move_dir_to_dir()
Christian Heimesada8c3b2008-03-18 18:26:33 +00001593
Serhiy Storchaka3a308b92014-02-11 10:30:59 +02001594 def test_move_dir_sep_to_dir(self):
1595 self._check_move_dir(self.src_dir + os.path.sep, self.dst_dir,
1596 os.path.join(self.dst_dir, os.path.basename(self.src_dir)))
1597
1598 @unittest.skipUnless(os.path.altsep, 'requires os.path.altsep')
1599 def test_move_dir_altsep_to_dir(self):
1600 self._check_move_dir(self.src_dir + os.path.altsep, self.dst_dir,
1601 os.path.join(self.dst_dir, os.path.basename(self.src_dir)))
1602
Christian Heimesada8c3b2008-03-18 18:26:33 +00001603 def test_existing_file_inside_dest_dir(self):
1604 # A file with the same name inside the destination dir already exists.
1605 with open(self.dst_file, "wb"):
1606 pass
1607 self.assertRaises(shutil.Error, shutil.move, self.src_file, self.dst_dir)
1608
1609 def test_dont_move_dir_in_itself(self):
1610 # Moving a dir inside itself raises an Error.
1611 dst = os.path.join(self.src_dir, "bar")
1612 self.assertRaises(shutil.Error, shutil.move, self.src_dir, dst)
1613
Antoine Pitrou0dcc3cd2009-01-29 20:26:59 +00001614 def test_destinsrc_false_negative(self):
1615 os.mkdir(TESTFN)
1616 try:
1617 for src, dst in [('srcdir', 'srcdir/dest')]:
1618 src = os.path.join(TESTFN, src)
1619 dst = os.path.join(TESTFN, dst)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001620 self.assertTrue(shutil._destinsrc(src, dst),
Benjamin Peterson247a9b82009-02-20 04:09:19 +00001621 msg='_destinsrc() wrongly concluded that '
Antoine Pitrou0dcc3cd2009-01-29 20:26:59 +00001622 'dst (%s) is not in src (%s)' % (dst, src))
1623 finally:
1624 shutil.rmtree(TESTFN, ignore_errors=True)
Christian Heimesada8c3b2008-03-18 18:26:33 +00001625
Antoine Pitrou0dcc3cd2009-01-29 20:26:59 +00001626 def test_destinsrc_false_positive(self):
1627 os.mkdir(TESTFN)
1628 try:
1629 for src, dst in [('srcdir', 'src/dest'), ('srcdir', 'srcdir.new')]:
1630 src = os.path.join(TESTFN, src)
1631 dst = os.path.join(TESTFN, dst)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001632 self.assertFalse(shutil._destinsrc(src, dst),
Benjamin Peterson247a9b82009-02-20 04:09:19 +00001633 msg='_destinsrc() wrongly concluded that '
Antoine Pitrou0dcc3cd2009-01-29 20:26:59 +00001634 'dst (%s) is in src (%s)' % (dst, src))
1635 finally:
1636 shutil.rmtree(TESTFN, ignore_errors=True)
Christian Heimes9bd667a2008-01-20 15:14:11 +00001637
Antoine Pitrou0a08d7a2012-01-06 20:16:19 +01001638 @support.skip_unless_symlink
1639 @mock_rename
1640 def test_move_file_symlink(self):
1641 dst = os.path.join(self.src_dir, 'bar')
1642 os.symlink(self.src_file, dst)
1643 shutil.move(dst, self.dst_file)
1644 self.assertTrue(os.path.islink(self.dst_file))
1645 self.assertTrue(os.path.samefile(self.src_file, self.dst_file))
1646
1647 @support.skip_unless_symlink
1648 @mock_rename
1649 def test_move_file_symlink_to_dir(self):
1650 filename = "bar"
1651 dst = os.path.join(self.src_dir, filename)
1652 os.symlink(self.src_file, dst)
1653 shutil.move(dst, self.dst_dir)
1654 final_link = os.path.join(self.dst_dir, filename)
1655 self.assertTrue(os.path.islink(final_link))
1656 self.assertTrue(os.path.samefile(self.src_file, final_link))
1657
1658 @support.skip_unless_symlink
1659 @mock_rename
1660 def test_move_dangling_symlink(self):
1661 src = os.path.join(self.src_dir, 'baz')
1662 dst = os.path.join(self.src_dir, 'bar')
1663 os.symlink(src, dst)
1664 dst_link = os.path.join(self.dst_dir, 'quux')
1665 shutil.move(dst, dst_link)
1666 self.assertTrue(os.path.islink(dst_link))
Antoine Pitrou3f48ac92014-01-01 02:50:45 +01001667 # On Windows, os.path.realpath does not follow symlinks (issue #9949)
1668 if os.name == 'nt':
1669 self.assertEqual(os.path.realpath(src), os.readlink(dst_link))
1670 else:
1671 self.assertEqual(os.path.realpath(src), os.path.realpath(dst_link))
Antoine Pitrou0a08d7a2012-01-06 20:16:19 +01001672
1673 @support.skip_unless_symlink
1674 @mock_rename
1675 def test_move_dir_symlink(self):
1676 src = os.path.join(self.src_dir, 'baz')
1677 dst = os.path.join(self.src_dir, 'bar')
1678 os.mkdir(src)
1679 os.symlink(src, dst)
1680 dst_link = os.path.join(self.dst_dir, 'quux')
1681 shutil.move(dst, dst_link)
1682 self.assertTrue(os.path.islink(dst_link))
1683 self.assertTrue(os.path.samefile(src, dst_link))
1684
Brian Curtin0d0a1de2012-06-18 18:41:07 -05001685 def test_move_return_value(self):
1686 rv = shutil.move(self.src_file, self.dst_dir)
1687 self.assertEqual(rv,
1688 os.path.join(self.dst_dir, os.path.basename(self.src_file)))
1689
1690 def test_move_as_rename_return_value(self):
1691 rv = shutil.move(self.src_file, os.path.join(self.dst_dir, 'bar'))
1692 self.assertEqual(rv, os.path.join(self.dst_dir, 'bar'))
1693
R David Murray6ffface2014-06-11 14:40:13 -04001694 @mock_rename
1695 def test_move_file_special_function(self):
1696 moved = []
1697 def _copy(src, dst):
1698 moved.append((src, dst))
1699 shutil.move(self.src_file, self.dst_dir, copy_function=_copy)
1700 self.assertEqual(len(moved), 1)
1701
1702 @mock_rename
1703 def test_move_dir_special_function(self):
1704 moved = []
1705 def _copy(src, dst):
1706 moved.append((src, dst))
1707 support.create_empty_file(os.path.join(self.src_dir, 'child'))
1708 support.create_empty_file(os.path.join(self.src_dir, 'child1'))
1709 shutil.move(self.src_dir, self.dst_dir, copy_function=_copy)
1710 self.assertEqual(len(moved), 3)
1711
Tarek Ziadé5340db32010-04-19 22:30:51 +00001712
Tarek Ziadéae4d5c62010-05-05 22:27:31 +00001713class TestCopyFile(unittest.TestCase):
1714
1715 _delete = False
1716
1717 class Faux(object):
1718 _entered = False
1719 _exited_with = None
1720 _raised = False
1721 def __init__(self, raise_in_exit=False, suppress_at_exit=True):
1722 self._raise_in_exit = raise_in_exit
1723 self._suppress_at_exit = suppress_at_exit
1724 def read(self, *args):
1725 return ''
1726 def __enter__(self):
1727 self._entered = True
1728 def __exit__(self, exc_type, exc_val, exc_tb):
1729 self._exited_with = exc_type, exc_val, exc_tb
1730 if self._raise_in_exit:
1731 self._raised = True
Andrew Svetlovf7a17b42012-12-25 16:47:37 +02001732 raise OSError("Cannot close")
Tarek Ziadéae4d5c62010-05-05 22:27:31 +00001733 return self._suppress_at_exit
1734
1735 def tearDown(self):
1736 if self._delete:
1737 del shutil.open
1738
1739 def _set_shutil_open(self, func):
1740 shutil.open = func
1741 self._delete = True
1742
1743 def test_w_source_open_fails(self):
1744 def _open(filename, mode='r'):
1745 if filename == 'srcfile':
Andrew Svetlovf7a17b42012-12-25 16:47:37 +02001746 raise OSError('Cannot open "srcfile"')
Tarek Ziadéae4d5c62010-05-05 22:27:31 +00001747 assert 0 # shouldn't reach here.
1748
1749 self._set_shutil_open(_open)
1750
Andrew Svetlovf7a17b42012-12-25 16:47:37 +02001751 self.assertRaises(OSError, shutil.copyfile, 'srcfile', 'destfile')
Tarek Ziadéae4d5c62010-05-05 22:27:31 +00001752
1753 def test_w_dest_open_fails(self):
1754
1755 srcfile = self.Faux()
1756
1757 def _open(filename, mode='r'):
1758 if filename == 'srcfile':
1759 return srcfile
1760 if filename == 'destfile':
Andrew Svetlovf7a17b42012-12-25 16:47:37 +02001761 raise OSError('Cannot open "destfile"')
Tarek Ziadéae4d5c62010-05-05 22:27:31 +00001762 assert 0 # shouldn't reach here.
1763
1764 self._set_shutil_open(_open)
1765
1766 shutil.copyfile('srcfile', 'destfile')
1767 self.assertTrue(srcfile._entered)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +02001768 self.assertTrue(srcfile._exited_with[0] is OSError)
Tarek Ziadéae4d5c62010-05-05 22:27:31 +00001769 self.assertEqual(srcfile._exited_with[1].args,
1770 ('Cannot open "destfile"',))
1771
1772 def test_w_dest_close_fails(self):
1773
1774 srcfile = self.Faux()
1775 destfile = self.Faux(True)
1776
1777 def _open(filename, mode='r'):
1778 if filename == 'srcfile':
1779 return srcfile
1780 if filename == 'destfile':
1781 return destfile
1782 assert 0 # shouldn't reach here.
1783
1784 self._set_shutil_open(_open)
1785
1786 shutil.copyfile('srcfile', 'destfile')
1787 self.assertTrue(srcfile._entered)
1788 self.assertTrue(destfile._entered)
1789 self.assertTrue(destfile._raised)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +02001790 self.assertTrue(srcfile._exited_with[0] is OSError)
Tarek Ziadéae4d5c62010-05-05 22:27:31 +00001791 self.assertEqual(srcfile._exited_with[1].args,
1792 ('Cannot close',))
1793
1794 def test_w_source_close_fails(self):
1795
1796 srcfile = self.Faux(True)
1797 destfile = self.Faux()
1798
1799 def _open(filename, mode='r'):
1800 if filename == 'srcfile':
1801 return srcfile
1802 if filename == 'destfile':
1803 return destfile
1804 assert 0 # shouldn't reach here.
1805
1806 self._set_shutil_open(_open)
1807
Andrew Svetlovf7a17b42012-12-25 16:47:37 +02001808 self.assertRaises(OSError,
Tarek Ziadéae4d5c62010-05-05 22:27:31 +00001809 shutil.copyfile, 'srcfile', 'destfile')
1810 self.assertTrue(srcfile._entered)
1811 self.assertTrue(destfile._entered)
1812 self.assertFalse(destfile._raised)
1813 self.assertTrue(srcfile._exited_with[0] is None)
1814 self.assertTrue(srcfile._raised)
1815
Ronald Oussorenf51738b2011-05-06 10:23:04 +02001816 def test_move_dir_caseinsensitive(self):
1817 # Renames a folder to the same name
1818 # but a different case.
1819
1820 self.src_dir = tempfile.mkdtemp()
Larry Hastings5b2f9c02012-06-25 23:50:01 -07001821 self.addCleanup(shutil.rmtree, self.src_dir, True)
Ronald Oussorenf51738b2011-05-06 10:23:04 +02001822 dst_dir = os.path.join(
1823 os.path.dirname(self.src_dir),
1824 os.path.basename(self.src_dir).upper())
1825 self.assertNotEqual(self.src_dir, dst_dir)
1826
1827 try:
1828 shutil.move(self.src_dir, dst_dir)
1829 self.assertTrue(os.path.isdir(dst_dir))
1830 finally:
Éric Araujoa7e33a12011-08-12 19:51:35 +02001831 os.rmdir(dst_dir)
Ronald Oussorenf51738b2011-05-06 10:23:04 +02001832
Antoine Pitroubcf2b592012-02-08 23:28:36 +01001833class TermsizeTests(unittest.TestCase):
1834 def test_does_not_crash(self):
1835 """Check if get_terminal_size() returns a meaningful value.
1836
1837 There's no easy portable way to actually check the size of the
1838 terminal, so let's check if it returns something sensible instead.
1839 """
1840 size = shutil.get_terminal_size()
Antoine Pitroucfade362012-02-08 23:48:59 +01001841 self.assertGreaterEqual(size.columns, 0)
1842 self.assertGreaterEqual(size.lines, 0)
Antoine Pitroubcf2b592012-02-08 23:28:36 +01001843
1844 def test_os_environ_first(self):
1845 "Check if environment variables have precedence"
1846
1847 with support.EnvironmentVarGuard() as env:
1848 env['COLUMNS'] = '777'
Serhiy Storchakad30829d2016-04-24 09:58:43 +03001849 del env['LINES']
Antoine Pitroubcf2b592012-02-08 23:28:36 +01001850 size = shutil.get_terminal_size()
1851 self.assertEqual(size.columns, 777)
1852
1853 with support.EnvironmentVarGuard() as env:
Serhiy Storchakad30829d2016-04-24 09:58:43 +03001854 del env['COLUMNS']
Antoine Pitroubcf2b592012-02-08 23:28:36 +01001855 env['LINES'] = '888'
1856 size = shutil.get_terminal_size()
1857 self.assertEqual(size.lines, 888)
1858
Serhiy Storchakad30829d2016-04-24 09:58:43 +03001859 def test_bad_environ(self):
1860 with support.EnvironmentVarGuard() as env:
1861 env['COLUMNS'] = 'xxx'
1862 env['LINES'] = 'yyy'
1863 size = shutil.get_terminal_size()
1864 self.assertGreaterEqual(size.columns, 0)
1865 self.assertGreaterEqual(size.lines, 0)
1866
Antoine Pitroubcf2b592012-02-08 23:28:36 +01001867 @unittest.skipUnless(os.isatty(sys.__stdout__.fileno()), "not on tty")
Victor Stinner119ebb72016-04-19 22:24:56 +02001868 @unittest.skipUnless(hasattr(os, 'get_terminal_size'),
1869 'need os.get_terminal_size()')
Antoine Pitroubcf2b592012-02-08 23:28:36 +01001870 def test_stty_match(self):
1871 """Check if stty returns the same results ignoring env
1872
1873 This test will fail if stdin and stdout are connected to
1874 different terminals with different sizes. Nevertheless, such
1875 situations should be pretty rare.
1876 """
1877 try:
1878 size = subprocess.check_output(['stty', 'size']).decode().split()
Xavier de Gaye38c8b7d2016-11-14 17:14:42 +01001879 except (FileNotFoundError, PermissionError,
1880 subprocess.CalledProcessError):
Antoine Pitroubcf2b592012-02-08 23:28:36 +01001881 self.skipTest("stty invocation failed")
1882 expected = (int(size[1]), int(size[0])) # reversed order
1883
1884 with support.EnvironmentVarGuard() as env:
1885 del env['LINES']
1886 del env['COLUMNS']
1887 actual = shutil.get_terminal_size()
1888
1889 self.assertEqual(expected, actual)
Ronald Oussorenf51738b2011-05-06 10:23:04 +02001890
Serhiy Storchakad30829d2016-04-24 09:58:43 +03001891 def test_fallback(self):
1892 with support.EnvironmentVarGuard() as env:
1893 del env['LINES']
1894 del env['COLUMNS']
1895
1896 # sys.__stdout__ has no fileno()
1897 with support.swap_attr(sys, '__stdout__', None):
1898 size = shutil.get_terminal_size(fallback=(10, 20))
1899 self.assertEqual(size.columns, 10)
1900 self.assertEqual(size.lines, 20)
1901
1902 # sys.__stdout__ is not a terminal on Unix
1903 # or fileno() not in (0, 1, 2) on Windows
1904 with open(os.devnull, 'w') as f, \
1905 support.swap_attr(sys, '__stdout__', f):
1906 size = shutil.get_terminal_size(fallback=(30, 40))
1907 self.assertEqual(size.columns, 30)
1908 self.assertEqual(size.lines, 40)
1909
Tarek Ziadéae4d5c62010-05-05 22:27:31 +00001910
Berker Peksag8083cd62014-11-01 11:04:06 +02001911class PublicAPITests(unittest.TestCase):
1912 """Ensures that the correct values are exposed in the public API."""
1913
1914 def test_module_all_attribute(self):
1915 self.assertTrue(hasattr(shutil, '__all__'))
1916 target_api = ['copyfileobj', 'copyfile', 'copymode', 'copystat',
1917 'copy', 'copy2', 'copytree', 'move', 'rmtree', 'Error',
1918 'SpecialFileError', 'ExecError', 'make_archive',
1919 'get_archive_formats', 'register_archive_format',
1920 'unregister_archive_format', 'get_unpack_formats',
1921 'register_unpack_format', 'unregister_unpack_format',
1922 'unpack_archive', 'ignore_patterns', 'chown', 'which',
1923 'get_terminal_size', 'SameFileError']
1924 if hasattr(os, 'statvfs') or os.name == 'nt':
1925 target_api.append('disk_usage')
1926 self.assertEqual(set(shutil.__all__), set(target_api))
1927
1928
Barry Warsaw7fc2cca2003-01-24 17:34:13 +00001929if __name__ == '__main__':
Brett Cannon3e9a9ae2013-06-12 21:25:59 -04001930 unittest.main()