summaryrefslogtreecommitdiffstats
path: root/Lib/os.py
diff options
context:
space:
mode:
authorSerhiy Storchaka <storchaka@gmail.com>2017-03-30 06:12:31 (GMT)
committerGitHub <noreply@github.com>2017-03-30 06:12:31 (GMT)
commitea720fe7e99d68924deab38de955fe97f87e2b29 (patch)
treea5996c9c8577f8824876575c8dcbf912d86226bc /Lib/os.py
parent0a58f72762353768c7d26412e627ff196aac6c4e (diff)
downloadcpython-ea720fe7e99d68924deab38de955fe97f87e2b29.zip
cpython-ea720fe7e99d68924deab38de955fe97f87e2b29.tar.gz
cpython-ea720fe7e99d68924deab38de955fe97f87e2b29.tar.bz2
bpo-25996: Added support of file descriptors in os.scandir() on Unix. (#502)
os.fwalk() is sped up by 2 times by using os.scandir().
Diffstat (limited to 'Lib/os.py')
-rw-r--r--Lib/os.py42
1 files changed, 25 insertions, 17 deletions
diff --git a/Lib/os.py b/Lib/os.py
index 70857c7..e293eca 100644
--- a/Lib/os.py
+++ b/Lib/os.py
@@ -129,6 +129,7 @@ if _exists("_have_functions"):
_add("HAVE_FCHMOD", "chmod")
_add("HAVE_FCHOWN", "chown")
_add("HAVE_FDOPENDIR", "listdir")
+ _add("HAVE_FDOPENDIR", "scandir")
_add("HAVE_FEXECVE", "execve")
_set.add(stat) # fstat always works
_add("HAVE_FTRUNCATE", "truncate")
@@ -416,7 +417,7 @@ def walk(top, topdown=True, onerror=None, followlinks=False):
__all__.append("walk")
-if {open, stat} <= supports_dir_fd and {listdir, stat} <= supports_fd:
+if {open, stat} <= supports_dir_fd and {scandir, stat} <= supports_fd:
def fwalk(top=".", topdown=True, onerror=None, *, follow_symlinks=False, dir_fd=None):
"""Directory tree generator.
@@ -455,7 +456,8 @@ if {open, stat} <= supports_dir_fd and {listdir, stat} <= supports_fd:
top = fspath(top)
# Note: To guard against symlink races, we use the standard
# lstat()/open()/fstat() trick.
- orig_st = stat(top, follow_symlinks=False, dir_fd=dir_fd)
+ if not follow_symlinks:
+ orig_st = stat(top, follow_symlinks=False, dir_fd=dir_fd)
topfd = open(top, O_RDONLY, dir_fd=dir_fd)
try:
if (follow_symlinks or (st.S_ISDIR(orig_st.st_mode) and
@@ -470,35 +472,41 @@ if {open, stat} <= supports_dir_fd and {listdir, stat} <= supports_fd:
# necessary, it can be adapted to only require O(1) FDs, see issue
# #13734.
- names = listdir(topfd)
- if isbytes:
- names = map(fsencode, names)
- dirs, nondirs = [], []
- for name in names:
+ scandir_it = scandir(topfd)
+ dirs = []
+ nondirs = []
+ entries = None if topdown or follow_symlinks else []
+ for entry in scandir_it:
+ name = entry.name
+ if isbytes:
+ name = fsencode(name)
try:
- # Here, we don't use AT_SYMLINK_NOFOLLOW to be consistent with
- # walk() which reports symlinks to directories as directories.
- # We do however check for symlinks before recursing into
- # a subdirectory.
- if st.S_ISDIR(stat(name, dir_fd=topfd).st_mode):
+ if entry.is_dir():
dirs.append(name)
+ if entries is not None:
+ entries.append(entry)
else:
nondirs.append(name)
except OSError:
try:
# Add dangling symlinks, ignore disappeared files
- if st.S_ISLNK(stat(name, dir_fd=topfd, follow_symlinks=False)
- .st_mode):
+ if entry.is_symlink():
nondirs.append(name)
except OSError:
- continue
+ pass
if topdown:
yield toppath, dirs, nondirs, topfd
- for name in dirs:
+ for name in dirs if entries is None else zip(dirs, entries):
try:
- orig_st = stat(name, dir_fd=topfd, follow_symlinks=follow_symlinks)
+ if not follow_symlinks:
+ if topdown:
+ orig_st = stat(name, dir_fd=topfd, follow_symlinks=False)
+ else:
+ assert entries is not None
+ name, entry = name
+ orig_st = entry.stat(follow_symlinks=False)
dirfd = open(name, O_RDONLY, dir_fd=topfd)
except OSError as err:
if onerror is not None: