Mercurial > repos > guerler > springsuite
comparison planemo/lib/python3.7/site-packages/scandir.py @ 1:56ad4e20f292 draft
"planemo upload commit 6eee67778febed82ddd413c3ca40b3183a3898f1"
author | guerler |
---|---|
date | Fri, 31 Jul 2020 00:32:28 -0400 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
0:d30785e31577 | 1:56ad4e20f292 |
---|---|
1 """scandir, a better directory iterator and faster os.walk(), now in the Python 3.5 stdlib | |
2 | |
3 scandir() is a generator version of os.listdir() that returns an | |
4 iterator over files in a directory, and also exposes the extra | |
5 information most OSes provide while iterating files in a directory | |
6 (such as type and stat information). | |
7 | |
8 This module also includes a version of os.walk() that uses scandir() | |
9 to speed it up significantly. | |
10 | |
11 See README.md or https://github.com/benhoyt/scandir for rationale and | |
12 docs, or read PEP 471 (https://www.python.org/dev/peps/pep-0471/) for | |
13 more details on its inclusion into Python 3.5 | |
14 | |
15 scandir is released under the new BSD 3-clause license. See | |
16 LICENSE.txt for the full license text. | |
17 """ | |
18 | |
19 from __future__ import division | |
20 | |
21 from errno import ENOENT | |
22 from os import listdir, lstat, stat, strerror | |
23 from os.path import join, islink | |
24 from stat import S_IFDIR, S_IFLNK, S_IFREG | |
25 import collections | |
26 import sys | |
27 | |
28 try: | |
29 import _scandir | |
30 except ImportError: | |
31 _scandir = None | |
32 | |
33 try: | |
34 import ctypes | |
35 except ImportError: | |
36 ctypes = None | |
37 | |
38 if _scandir is None and ctypes is None: | |
39 import warnings | |
40 warnings.warn("scandir can't find the compiled _scandir C module " | |
41 "or ctypes, using slow generic fallback") | |
42 | |
43 __version__ = '1.10.0' | |
44 __all__ = ['scandir', 'walk'] | |
45 | |
46 # Windows FILE_ATTRIBUTE constants for interpreting the | |
47 # FIND_DATA.dwFileAttributes member | |
48 FILE_ATTRIBUTE_ARCHIVE = 32 | |
49 FILE_ATTRIBUTE_COMPRESSED = 2048 | |
50 FILE_ATTRIBUTE_DEVICE = 64 | |
51 FILE_ATTRIBUTE_DIRECTORY = 16 | |
52 FILE_ATTRIBUTE_ENCRYPTED = 16384 | |
53 FILE_ATTRIBUTE_HIDDEN = 2 | |
54 FILE_ATTRIBUTE_INTEGRITY_STREAM = 32768 | |
55 FILE_ATTRIBUTE_NORMAL = 128 | |
56 FILE_ATTRIBUTE_NOT_CONTENT_INDEXED = 8192 | |
57 FILE_ATTRIBUTE_NO_SCRUB_DATA = 131072 | |
58 FILE_ATTRIBUTE_OFFLINE = 4096 | |
59 FILE_ATTRIBUTE_READONLY = 1 | |
60 FILE_ATTRIBUTE_REPARSE_POINT = 1024 | |
61 FILE_ATTRIBUTE_SPARSE_FILE = 512 | |
62 FILE_ATTRIBUTE_SYSTEM = 4 | |
63 FILE_ATTRIBUTE_TEMPORARY = 256 | |
64 FILE_ATTRIBUTE_VIRTUAL = 65536 | |
65 | |
66 IS_PY3 = sys.version_info >= (3, 0) | |
67 | |
68 if IS_PY3: | |
69 unicode = str # Because Python <= 3.2 doesn't have u'unicode' syntax | |
70 | |
71 | |
72 class GenericDirEntry(object): | |
73 __slots__ = ('name', '_stat', '_lstat', '_scandir_path', '_path') | |
74 | |
75 def __init__(self, scandir_path, name): | |
76 self._scandir_path = scandir_path | |
77 self.name = name | |
78 self._stat = None | |
79 self._lstat = None | |
80 self._path = None | |
81 | |
82 @property | |
83 def path(self): | |
84 if self._path is None: | |
85 self._path = join(self._scandir_path, self.name) | |
86 return self._path | |
87 | |
88 def stat(self, follow_symlinks=True): | |
89 if follow_symlinks: | |
90 if self._stat is None: | |
91 self._stat = stat(self.path) | |
92 return self._stat | |
93 else: | |
94 if self._lstat is None: | |
95 self._lstat = lstat(self.path) | |
96 return self._lstat | |
97 | |
98 # The code duplication below is intentional: this is for slightly | |
99 # better performance on systems that fall back to GenericDirEntry. | |
100 # It avoids an additional attribute lookup and method call, which | |
101 # are relatively slow on CPython. | |
102 def is_dir(self, follow_symlinks=True): | |
103 try: | |
104 st = self.stat(follow_symlinks=follow_symlinks) | |
105 except OSError as e: | |
106 if e.errno != ENOENT: | |
107 raise | |
108 return False # Path doesn't exist or is a broken symlink | |
109 return st.st_mode & 0o170000 == S_IFDIR | |
110 | |
111 def is_file(self, follow_symlinks=True): | |
112 try: | |
113 st = self.stat(follow_symlinks=follow_symlinks) | |
114 except OSError as e: | |
115 if e.errno != ENOENT: | |
116 raise | |
117 return False # Path doesn't exist or is a broken symlink | |
118 return st.st_mode & 0o170000 == S_IFREG | |
119 | |
120 def is_symlink(self): | |
121 try: | |
122 st = self.stat(follow_symlinks=False) | |
123 except OSError as e: | |
124 if e.errno != ENOENT: | |
125 raise | |
126 return False # Path doesn't exist or is a broken symlink | |
127 return st.st_mode & 0o170000 == S_IFLNK | |
128 | |
129 def inode(self): | |
130 st = self.stat(follow_symlinks=False) | |
131 return st.st_ino | |
132 | |
133 def __str__(self): | |
134 return '<{0}: {1!r}>'.format(self.__class__.__name__, self.name) | |
135 | |
136 __repr__ = __str__ | |
137 | |
138 | |
139 def _scandir_generic(path=unicode('.')): | |
140 """Like os.listdir(), but yield DirEntry objects instead of returning | |
141 a list of names. | |
142 """ | |
143 for name in listdir(path): | |
144 yield GenericDirEntry(path, name) | |
145 | |
146 | |
147 if IS_PY3 and sys.platform == 'win32': | |
148 def scandir_generic(path=unicode('.')): | |
149 if isinstance(path, bytes): | |
150 raise TypeError("os.scandir() doesn't support bytes path on Windows, use Unicode instead") | |
151 return _scandir_generic(path) | |
152 scandir_generic.__doc__ = _scandir_generic.__doc__ | |
153 else: | |
154 scandir_generic = _scandir_generic | |
155 | |
156 | |
157 scandir_c = None | |
158 scandir_python = None | |
159 | |
160 | |
161 if sys.platform == 'win32': | |
162 if ctypes is not None: | |
163 from ctypes import wintypes | |
164 | |
165 # Various constants from windows.h | |
166 INVALID_HANDLE_VALUE = ctypes.c_void_p(-1).value | |
167 ERROR_FILE_NOT_FOUND = 2 | |
168 ERROR_NO_MORE_FILES = 18 | |
169 IO_REPARSE_TAG_SYMLINK = 0xA000000C | |
170 | |
171 # Numer of seconds between 1601-01-01 and 1970-01-01 | |
172 SECONDS_BETWEEN_EPOCHS = 11644473600 | |
173 | |
174 kernel32 = ctypes.windll.kernel32 | |
175 | |
176 # ctypes wrappers for (wide string versions of) FindFirstFile, | |
177 # FindNextFile, and FindClose | |
178 FindFirstFile = kernel32.FindFirstFileW | |
179 FindFirstFile.argtypes = [ | |
180 wintypes.LPCWSTR, | |
181 ctypes.POINTER(wintypes.WIN32_FIND_DATAW), | |
182 ] | |
183 FindFirstFile.restype = wintypes.HANDLE | |
184 | |
185 FindNextFile = kernel32.FindNextFileW | |
186 FindNextFile.argtypes = [ | |
187 wintypes.HANDLE, | |
188 ctypes.POINTER(wintypes.WIN32_FIND_DATAW), | |
189 ] | |
190 FindNextFile.restype = wintypes.BOOL | |
191 | |
192 FindClose = kernel32.FindClose | |
193 FindClose.argtypes = [wintypes.HANDLE] | |
194 FindClose.restype = wintypes.BOOL | |
195 | |
196 Win32StatResult = collections.namedtuple('Win32StatResult', [ | |
197 'st_mode', | |
198 'st_ino', | |
199 'st_dev', | |
200 'st_nlink', | |
201 'st_uid', | |
202 'st_gid', | |
203 'st_size', | |
204 'st_atime', | |
205 'st_mtime', | |
206 'st_ctime', | |
207 'st_atime_ns', | |
208 'st_mtime_ns', | |
209 'st_ctime_ns', | |
210 'st_file_attributes', | |
211 ]) | |
212 | |
213 def filetime_to_time(filetime): | |
214 """Convert Win32 FILETIME to time since Unix epoch in seconds.""" | |
215 total = filetime.dwHighDateTime << 32 | filetime.dwLowDateTime | |
216 return total / 10000000 - SECONDS_BETWEEN_EPOCHS | |
217 | |
218 def find_data_to_stat(data): | |
219 """Convert Win32 FIND_DATA struct to stat_result.""" | |
220 # First convert Win32 dwFileAttributes to st_mode | |
221 attributes = data.dwFileAttributes | |
222 st_mode = 0 | |
223 if attributes & FILE_ATTRIBUTE_DIRECTORY: | |
224 st_mode |= S_IFDIR | 0o111 | |
225 else: | |
226 st_mode |= S_IFREG | |
227 if attributes & FILE_ATTRIBUTE_READONLY: | |
228 st_mode |= 0o444 | |
229 else: | |
230 st_mode |= 0o666 | |
231 if (attributes & FILE_ATTRIBUTE_REPARSE_POINT and | |
232 data.dwReserved0 == IO_REPARSE_TAG_SYMLINK): | |
233 st_mode ^= st_mode & 0o170000 | |
234 st_mode |= S_IFLNK | |
235 | |
236 st_size = data.nFileSizeHigh << 32 | data.nFileSizeLow | |
237 st_atime = filetime_to_time(data.ftLastAccessTime) | |
238 st_mtime = filetime_to_time(data.ftLastWriteTime) | |
239 st_ctime = filetime_to_time(data.ftCreationTime) | |
240 | |
241 # Some fields set to zero per CPython's posixmodule.c: st_ino, st_dev, | |
242 # st_nlink, st_uid, st_gid | |
243 return Win32StatResult(st_mode, 0, 0, 0, 0, 0, st_size, | |
244 st_atime, st_mtime, st_ctime, | |
245 int(st_atime * 1000000000), | |
246 int(st_mtime * 1000000000), | |
247 int(st_ctime * 1000000000), | |
248 attributes) | |
249 | |
250 class Win32DirEntryPython(object): | |
251 __slots__ = ('name', '_stat', '_lstat', '_find_data', '_scandir_path', '_path', '_inode') | |
252 | |
253 def __init__(self, scandir_path, name, find_data): | |
254 self._scandir_path = scandir_path | |
255 self.name = name | |
256 self._stat = None | |
257 self._lstat = None | |
258 self._find_data = find_data | |
259 self._path = None | |
260 self._inode = None | |
261 | |
262 @property | |
263 def path(self): | |
264 if self._path is None: | |
265 self._path = join(self._scandir_path, self.name) | |
266 return self._path | |
267 | |
268 def stat(self, follow_symlinks=True): | |
269 if follow_symlinks: | |
270 if self._stat is None: | |
271 if self.is_symlink(): | |
272 # It's a symlink, call link-following stat() | |
273 self._stat = stat(self.path) | |
274 else: | |
275 # Not a symlink, stat is same as lstat value | |
276 if self._lstat is None: | |
277 self._lstat = find_data_to_stat(self._find_data) | |
278 self._stat = self._lstat | |
279 return self._stat | |
280 else: | |
281 if self._lstat is None: | |
282 # Lazily convert to stat object, because it's slow | |
283 # in Python, and often we only need is_dir() etc | |
284 self._lstat = find_data_to_stat(self._find_data) | |
285 return self._lstat | |
286 | |
287 def is_dir(self, follow_symlinks=True): | |
288 is_symlink = self.is_symlink() | |
289 if follow_symlinks and is_symlink: | |
290 try: | |
291 return self.stat().st_mode & 0o170000 == S_IFDIR | |
292 except OSError as e: | |
293 if e.errno != ENOENT: | |
294 raise | |
295 return False | |
296 elif is_symlink: | |
297 return False | |
298 else: | |
299 return (self._find_data.dwFileAttributes & | |
300 FILE_ATTRIBUTE_DIRECTORY != 0) | |
301 | |
302 def is_file(self, follow_symlinks=True): | |
303 is_symlink = self.is_symlink() | |
304 if follow_symlinks and is_symlink: | |
305 try: | |
306 return self.stat().st_mode & 0o170000 == S_IFREG | |
307 except OSError as e: | |
308 if e.errno != ENOENT: | |
309 raise | |
310 return False | |
311 elif is_symlink: | |
312 return False | |
313 else: | |
314 return (self._find_data.dwFileAttributes & | |
315 FILE_ATTRIBUTE_DIRECTORY == 0) | |
316 | |
317 def is_symlink(self): | |
318 return (self._find_data.dwFileAttributes & | |
319 FILE_ATTRIBUTE_REPARSE_POINT != 0 and | |
320 self._find_data.dwReserved0 == IO_REPARSE_TAG_SYMLINK) | |
321 | |
322 def inode(self): | |
323 if self._inode is None: | |
324 self._inode = lstat(self.path).st_ino | |
325 return self._inode | |
326 | |
327 def __str__(self): | |
328 return '<{0}: {1!r}>'.format(self.__class__.__name__, self.name) | |
329 | |
330 __repr__ = __str__ | |
331 | |
332 def win_error(error, filename): | |
333 exc = WindowsError(error, ctypes.FormatError(error)) | |
334 exc.filename = filename | |
335 return exc | |
336 | |
337 def _scandir_python(path=unicode('.')): | |
338 """Like os.listdir(), but yield DirEntry objects instead of returning | |
339 a list of names. | |
340 """ | |
341 # Call FindFirstFile and handle errors | |
342 if isinstance(path, bytes): | |
343 is_bytes = True | |
344 filename = join(path.decode('mbcs', 'strict'), '*.*') | |
345 else: | |
346 is_bytes = False | |
347 filename = join(path, '*.*') | |
348 data = wintypes.WIN32_FIND_DATAW() | |
349 data_p = ctypes.byref(data) | |
350 handle = FindFirstFile(filename, data_p) | |
351 if handle == INVALID_HANDLE_VALUE: | |
352 error = ctypes.GetLastError() | |
353 if error == ERROR_FILE_NOT_FOUND: | |
354 # No files, don't yield anything | |
355 return | |
356 raise win_error(error, path) | |
357 | |
358 # Call FindNextFile in a loop, stopping when no more files | |
359 try: | |
360 while True: | |
361 # Skip '.' and '..' (current and parent directory), but | |
362 # otherwise yield (filename, stat_result) tuple | |
363 name = data.cFileName | |
364 if name not in ('.', '..'): | |
365 if is_bytes: | |
366 name = name.encode('mbcs', 'replace') | |
367 yield Win32DirEntryPython(path, name, data) | |
368 | |
369 data = wintypes.WIN32_FIND_DATAW() | |
370 data_p = ctypes.byref(data) | |
371 success = FindNextFile(handle, data_p) | |
372 if not success: | |
373 error = ctypes.GetLastError() | |
374 if error == ERROR_NO_MORE_FILES: | |
375 break | |
376 raise win_error(error, path) | |
377 finally: | |
378 if not FindClose(handle): | |
379 raise win_error(ctypes.GetLastError(), path) | |
380 | |
381 if IS_PY3: | |
382 def scandir_python(path=unicode('.')): | |
383 if isinstance(path, bytes): | |
384 raise TypeError("os.scandir() doesn't support bytes path on Windows, use Unicode instead") | |
385 return _scandir_python(path) | |
386 scandir_python.__doc__ = _scandir_python.__doc__ | |
387 else: | |
388 scandir_python = _scandir_python | |
389 | |
390 if _scandir is not None: | |
391 scandir_c = _scandir.scandir | |
392 DirEntry_c = _scandir.DirEntry | |
393 | |
394 if _scandir is not None: | |
395 scandir = scandir_c | |
396 DirEntry = DirEntry_c | |
397 elif ctypes is not None: | |
398 scandir = scandir_python | |
399 DirEntry = Win32DirEntryPython | |
400 else: | |
401 scandir = scandir_generic | |
402 DirEntry = GenericDirEntry | |
403 | |
404 | |
405 # Linux, OS X, and BSD implementation | |
406 elif sys.platform.startswith(('linux', 'darwin', 'sunos5')) or 'bsd' in sys.platform: | |
407 have_dirent_d_type = (sys.platform != 'sunos5') | |
408 | |
409 if ctypes is not None and have_dirent_d_type: | |
410 import ctypes.util | |
411 | |
412 DIR_p = ctypes.c_void_p | |
413 | |
414 # Rather annoying how the dirent struct is slightly different on each | |
415 # platform. The only fields we care about are d_name and d_type. | |
416 class Dirent(ctypes.Structure): | |
417 if sys.platform.startswith('linux'): | |
418 _fields_ = ( | |
419 ('d_ino', ctypes.c_ulong), | |
420 ('d_off', ctypes.c_long), | |
421 ('d_reclen', ctypes.c_ushort), | |
422 ('d_type', ctypes.c_byte), | |
423 ('d_name', ctypes.c_char * 256), | |
424 ) | |
425 elif 'openbsd' in sys.platform: | |
426 _fields_ = ( | |
427 ('d_ino', ctypes.c_uint64), | |
428 ('d_off', ctypes.c_uint64), | |
429 ('d_reclen', ctypes.c_uint16), | |
430 ('d_type', ctypes.c_uint8), | |
431 ('d_namlen', ctypes.c_uint8), | |
432 ('__d_padding', ctypes.c_uint8 * 4), | |
433 ('d_name', ctypes.c_char * 256), | |
434 ) | |
435 else: | |
436 _fields_ = ( | |
437 ('d_ino', ctypes.c_uint32), # must be uint32, not ulong | |
438 ('d_reclen', ctypes.c_ushort), | |
439 ('d_type', ctypes.c_byte), | |
440 ('d_namlen', ctypes.c_byte), | |
441 ('d_name', ctypes.c_char * 256), | |
442 ) | |
443 | |
444 DT_UNKNOWN = 0 | |
445 DT_DIR = 4 | |
446 DT_REG = 8 | |
447 DT_LNK = 10 | |
448 | |
449 Dirent_p = ctypes.POINTER(Dirent) | |
450 Dirent_pp = ctypes.POINTER(Dirent_p) | |
451 | |
452 libc = ctypes.CDLL(ctypes.util.find_library('c'), use_errno=True) | |
453 opendir = libc.opendir | |
454 opendir.argtypes = [ctypes.c_char_p] | |
455 opendir.restype = DIR_p | |
456 | |
457 readdir_r = libc.readdir_r | |
458 readdir_r.argtypes = [DIR_p, Dirent_p, Dirent_pp] | |
459 readdir_r.restype = ctypes.c_int | |
460 | |
461 closedir = libc.closedir | |
462 closedir.argtypes = [DIR_p] | |
463 closedir.restype = ctypes.c_int | |
464 | |
465 file_system_encoding = sys.getfilesystemencoding() | |
466 | |
467 class PosixDirEntry(object): | |
468 __slots__ = ('name', '_d_type', '_stat', '_lstat', '_scandir_path', '_path', '_inode') | |
469 | |
470 def __init__(self, scandir_path, name, d_type, inode): | |
471 self._scandir_path = scandir_path | |
472 self.name = name | |
473 self._d_type = d_type | |
474 self._inode = inode | |
475 self._stat = None | |
476 self._lstat = None | |
477 self._path = None | |
478 | |
479 @property | |
480 def path(self): | |
481 if self._path is None: | |
482 self._path = join(self._scandir_path, self.name) | |
483 return self._path | |
484 | |
485 def stat(self, follow_symlinks=True): | |
486 if follow_symlinks: | |
487 if self._stat is None: | |
488 if self.is_symlink(): | |
489 self._stat = stat(self.path) | |
490 else: | |
491 if self._lstat is None: | |
492 self._lstat = lstat(self.path) | |
493 self._stat = self._lstat | |
494 return self._stat | |
495 else: | |
496 if self._lstat is None: | |
497 self._lstat = lstat(self.path) | |
498 return self._lstat | |
499 | |
500 def is_dir(self, follow_symlinks=True): | |
501 if (self._d_type == DT_UNKNOWN or | |
502 (follow_symlinks and self.is_symlink())): | |
503 try: | |
504 st = self.stat(follow_symlinks=follow_symlinks) | |
505 except OSError as e: | |
506 if e.errno != ENOENT: | |
507 raise | |
508 return False | |
509 return st.st_mode & 0o170000 == S_IFDIR | |
510 else: | |
511 return self._d_type == DT_DIR | |
512 | |
513 def is_file(self, follow_symlinks=True): | |
514 if (self._d_type == DT_UNKNOWN or | |
515 (follow_symlinks and self.is_symlink())): | |
516 try: | |
517 st = self.stat(follow_symlinks=follow_symlinks) | |
518 except OSError as e: | |
519 if e.errno != ENOENT: | |
520 raise | |
521 return False | |
522 return st.st_mode & 0o170000 == S_IFREG | |
523 else: | |
524 return self._d_type == DT_REG | |
525 | |
526 def is_symlink(self): | |
527 if self._d_type == DT_UNKNOWN: | |
528 try: | |
529 st = self.stat(follow_symlinks=False) | |
530 except OSError as e: | |
531 if e.errno != ENOENT: | |
532 raise | |
533 return False | |
534 return st.st_mode & 0o170000 == S_IFLNK | |
535 else: | |
536 return self._d_type == DT_LNK | |
537 | |
538 def inode(self): | |
539 return self._inode | |
540 | |
541 def __str__(self): | |
542 return '<{0}: {1!r}>'.format(self.__class__.__name__, self.name) | |
543 | |
544 __repr__ = __str__ | |
545 | |
546 def posix_error(filename): | |
547 errno = ctypes.get_errno() | |
548 exc = OSError(errno, strerror(errno)) | |
549 exc.filename = filename | |
550 return exc | |
551 | |
552 def scandir_python(path=unicode('.')): | |
553 """Like os.listdir(), but yield DirEntry objects instead of returning | |
554 a list of names. | |
555 """ | |
556 if isinstance(path, bytes): | |
557 opendir_path = path | |
558 is_bytes = True | |
559 else: | |
560 opendir_path = path.encode(file_system_encoding) | |
561 is_bytes = False | |
562 dir_p = opendir(opendir_path) | |
563 if not dir_p: | |
564 raise posix_error(path) | |
565 try: | |
566 result = Dirent_p() | |
567 while True: | |
568 entry = Dirent() | |
569 if readdir_r(dir_p, entry, result): | |
570 raise posix_error(path) | |
571 if not result: | |
572 break | |
573 name = entry.d_name | |
574 if name not in (b'.', b'..'): | |
575 if not is_bytes: | |
576 name = name.decode(file_system_encoding) | |
577 yield PosixDirEntry(path, name, entry.d_type, entry.d_ino) | |
578 finally: | |
579 if closedir(dir_p): | |
580 raise posix_error(path) | |
581 | |
582 if _scandir is not None: | |
583 scandir_c = _scandir.scandir | |
584 DirEntry_c = _scandir.DirEntry | |
585 | |
586 if _scandir is not None: | |
587 scandir = scandir_c | |
588 DirEntry = DirEntry_c | |
589 elif ctypes is not None and have_dirent_d_type: | |
590 scandir = scandir_python | |
591 DirEntry = PosixDirEntry | |
592 else: | |
593 scandir = scandir_generic | |
594 DirEntry = GenericDirEntry | |
595 | |
596 | |
597 # Some other system -- no d_type or stat information | |
598 else: | |
599 scandir = scandir_generic | |
600 DirEntry = GenericDirEntry | |
601 | |
602 | |
603 def _walk(top, topdown=True, onerror=None, followlinks=False): | |
604 """Like Python 3.5's implementation of os.walk() -- faster than | |
605 the pre-Python 3.5 version as it uses scandir() internally. | |
606 """ | |
607 dirs = [] | |
608 nondirs = [] | |
609 | |
610 # We may not have read permission for top, in which case we can't | |
611 # get a list of the files the directory contains. os.walk | |
612 # always suppressed the exception then, rather than blow up for a | |
613 # minor reason when (say) a thousand readable directories are still | |
614 # left to visit. That logic is copied here. | |
615 try: | |
616 scandir_it = scandir(top) | |
617 except OSError as error: | |
618 if onerror is not None: | |
619 onerror(error) | |
620 return | |
621 | |
622 while True: | |
623 try: | |
624 try: | |
625 entry = next(scandir_it) | |
626 except StopIteration: | |
627 break | |
628 except OSError as error: | |
629 if onerror is not None: | |
630 onerror(error) | |
631 return | |
632 | |
633 try: | |
634 is_dir = entry.is_dir() | |
635 except OSError: | |
636 # If is_dir() raises an OSError, consider that the entry is not | |
637 # a directory, same behaviour than os.path.isdir(). | |
638 is_dir = False | |
639 | |
640 if is_dir: | |
641 dirs.append(entry.name) | |
642 else: | |
643 nondirs.append(entry.name) | |
644 | |
645 if not topdown and is_dir: | |
646 # Bottom-up: recurse into sub-directory, but exclude symlinks to | |
647 # directories if followlinks is False | |
648 if followlinks: | |
649 walk_into = True | |
650 else: | |
651 try: | |
652 is_symlink = entry.is_symlink() | |
653 except OSError: | |
654 # If is_symlink() raises an OSError, consider that the | |
655 # entry is not a symbolic link, same behaviour than | |
656 # os.path.islink(). | |
657 is_symlink = False | |
658 walk_into = not is_symlink | |
659 | |
660 if walk_into: | |
661 for entry in walk(entry.path, topdown, onerror, followlinks): | |
662 yield entry | |
663 | |
664 # Yield before recursion if going top down | |
665 if topdown: | |
666 yield top, dirs, nondirs | |
667 | |
668 # Recurse into sub-directories | |
669 for name in dirs: | |
670 new_path = join(top, name) | |
671 # Issue #23605: os.path.islink() is used instead of caching | |
672 # entry.is_symlink() result during the loop on os.scandir() because | |
673 # the caller can replace the directory entry during the "yield" | |
674 # above. | |
675 if followlinks or not islink(new_path): | |
676 for entry in walk(new_path, topdown, onerror, followlinks): | |
677 yield entry | |
678 else: | |
679 # Yield after recursion if going bottom up | |
680 yield top, dirs, nondirs | |
681 | |
682 | |
683 if IS_PY3 or sys.platform != 'win32': | |
684 walk = _walk | |
685 else: | |
686 # Fix for broken unicode handling on Windows on Python 2.x, see: | |
687 # https://github.com/benhoyt/scandir/issues/54 | |
688 file_system_encoding = sys.getfilesystemencoding() | |
689 | |
690 def walk(top, topdown=True, onerror=None, followlinks=False): | |
691 if isinstance(top, bytes): | |
692 top = top.decode(file_system_encoding) | |
693 return _walk(top, topdown, onerror, followlinks) |