summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMiss Islington (bot) <31488909+miss-islington@users.noreply.github.com>2024-05-30 04:30:37 (GMT)
committerGitHub <noreply@github.com>2024-05-30 04:30:37 (GMT)
commitd4a146d56727f456e579419cbb2bb72e6667103a (patch)
tree32f2757e7db90af5e558cd1eb68af234318d1e98
parentaae371bda47d4c758059264f86a3ed6cbaa15a4a (diff)
downloadcpython-d4a146d56727f456e579419cbb2bb72e6667103a.zip
cpython-d4a146d56727f456e579419cbb2bb72e6667103a.tar.gz
cpython-d4a146d56727f456e579419cbb2bb72e6667103a.tar.bz2
[3.12] GH-89727: Fix FD leak on `os.fwalk()` generator finalization. (GH-119766) (#119768)
GH-89727: Fix FD leak on `os.fwalk()` generator finalization. (GH-119766) Follow-up to 3c890b50. Ensure we `os.close()` open file descriptors when the `os.fwalk()` generator is finalized. (cherry picked from commit a5fef800d31648d19cecc240a2fa0dc71371753e) Co-authored-by: Barney Gale <barney.gale@gmail.com>
-rw-r--r--Lib/os.py11
-rw-r--r--Lib/test/test_os.py21
2 files changed, 30 insertions, 2 deletions
diff --git a/Lib/os.py b/Lib/os.py
index 3e911c9..5619470 100644
--- a/Lib/os.py
+++ b/Lib/os.py
@@ -478,8 +478,15 @@ if {open, stat} <= supports_dir_fd and {scandir, stat} <= supports_fd:
top = fspath(top)
stack = [(_fwalk_walk, (True, dir_fd, top, top, None))]
isbytes = isinstance(top, bytes)
- while stack:
- yield from _fwalk(stack, isbytes, topdown, onerror, follow_symlinks)
+ try:
+ while stack:
+ yield from _fwalk(stack, isbytes, topdown, onerror, follow_symlinks)
+ finally:
+ # Close any file descriptors still on the stack.
+ while stack:
+ action, value = stack.pop()
+ if action == _fwalk_close:
+ close(value)
# Each item in the _fwalk() stack is a pair (action, args).
_fwalk_walk = 0 # args: (isroot, dirfd, toppath, topname, entry)
diff --git a/Lib/test/test_os.py b/Lib/test/test_os.py
index a764632..8803705 100644
--- a/Lib/test/test_os.py
+++ b/Lib/test/test_os.py
@@ -1671,6 +1671,27 @@ class FwalkTests(WalkTests):
self.addCleanup(os.close, newfd)
self.assertEqual(newfd, minfd)
+ @unittest.skipIf(
+ support.is_emscripten, "Cannot dup stdout on Emscripten"
+ )
+ @unittest.skipIf(
+ support.is_android, "dup return value is unpredictable on Android"
+ )
+ def test_fd_finalization(self):
+ # Check that close()ing the fwalk() generator closes FDs
+ def getfd():
+ fd = os.dup(1)
+ os.close(fd)
+ return fd
+ for topdown in (False, True):
+ old_fd = getfd()
+ it = self.fwalk(os_helper.TESTFN, topdown=topdown)
+ self.assertEqual(getfd(), old_fd)
+ next(it)
+ self.assertGreater(getfd(), old_fd)
+ it.close()
+ self.assertEqual(getfd(), old_fd)
+
# fwalk() keeps file descriptors open
test_walk_many_open_files = None