Skip to content

Commit 94c1b86

Browse files
committed
Add missing support for os.renames
- fixes #714
1 parent 905d337 commit 94c1b86

File tree

3 files changed

+49
-1
lines changed

3 files changed

+49
-1
lines changed

CHANGES.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ The released versions correspond to PyPi releases.
2121
is of type `byte`; the call itself does nothing as before
2222
* do not skip filesystem modules by name to allow using own modules with
2323
the same name (see [#707](../../issues/707))
24-
24+
* add missing support for `os.renames` (see [#714](../../issues/714))
2525

2626
## [Version 4.6.3](https://pypi.python.org/pypi/pyfakefs/4.6.3) (2022-07-20)
2727
Another patch release that fixes a regression in version 4.6.

pyfakefs/fake_filesystem.py

+26
Original file line numberDiff line numberDiff line change
@@ -4477,6 +4477,32 @@ def rename(self, src: AnyStr, dst: AnyStr, *,
44774477
dst = self._path_with_dir_fd(dst, self.rename, dst_dir_fd)
44784478
self.filesystem.rename(src, dst)
44794479

4480+
def renames(self, old: AnyStr, new: AnyStr):
4481+
"""Fakes `os.renames`, documentation taken from there.
4482+
4483+
Super-rename; create directories as necessary and delete any left
4484+
empty. Works like rename, except creation of any intermediate
4485+
directories needed to make the new pathname good is attempted
4486+
first. After the rename, directories corresponding to rightmost
4487+
path segments of the old name will be pruned until either the
4488+
whole path is consumed or a nonempty directory is found.
4489+
4490+
Note: this function can fail with the new directory structure made
4491+
if you lack permissions needed to unlink the leaf directory or
4492+
file.
4493+
4494+
"""
4495+
head, tail = self.filesystem.splitpath(new)
4496+
if head and tail and not self.filesystem.exists(head):
4497+
self.makedirs(head)
4498+
self.rename(old, new)
4499+
head, tail = self.filesystem.splitpath(old)
4500+
if head and tail:
4501+
try:
4502+
self.removedirs(head)
4503+
except OSError:
4504+
pass
4505+
44804506
def replace(self, src: AnyStr, dst: AnyStr, *,
44814507
src_dir_fd: Optional[int] = None,
44824508
dst_dir_fd: Optional[int] = None) -> None:

pyfakefs/tests/fake_os_test.py

+22
Original file line numberDiff line numberDiff line change
@@ -3328,6 +3328,28 @@ def test_rename_symlink_to_other_case_works_in_windows(self):
33283328
self.os.rename(path0, path1)
33293329
self.assertEqual(['Beta'], sorted(self.os.listdir(path0)))
33303330

3331+
def test_renames_creates_missing_dirs(self):
3332+
old_path = self.make_path("foo.txt")
3333+
self.create_file(old_path)
3334+
new_path = self.make_path("new", "dir", "bar.txt")
3335+
self.os.renames(old_path, new_path)
3336+
self.assertTrue(self.os.path.exists(new_path))
3337+
self.assertFalse(self.os.path.exists(old_path))
3338+
3339+
def test_renames_removes_empty_dirs(self):
3340+
old_base_path = self.make_path("old")
3341+
old_path = self.make_path("old", "dir1", "dir2", "foo.txt")
3342+
other_file = self.os.path.join(old_base_path, "foo.png")
3343+
self.create_file(old_path)
3344+
self.create_file(other_file)
3345+
new_path = self.make_path("new", "bar.txt")
3346+
self.os.renames(old_path, new_path)
3347+
self.assertTrue(self.os.path.exists(new_path))
3348+
self.assertFalse(self.os.path.exists(old_path))
3349+
self.assertTrue(self.os.path.exists(old_base_path))
3350+
removed_path = self.os.path.join(old_base_path, "dir1")
3351+
self.assertFalse(self.os.path.exists(removed_path))
3352+
33313353
def test_stat_with_mixed_case(self):
33323354
# Regression test for #310
33333355
self.skip_if_symlink_not_supported()

0 commit comments

Comments
 (0)