Mercurial > repos > guerler > springsuite
comparison planemo/lib/python3.7/site-packages/pathlib2/__init__.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 # Copyright (c) 2014-2017 Matthias C. M. Troffaes | |
2 # Copyright (c) 2012-2014 Antoine Pitrou and contributors | |
3 # Distributed under the terms of the MIT License. | |
4 | |
5 import ctypes | |
6 import fnmatch | |
7 import functools | |
8 import io | |
9 import ntpath | |
10 import os | |
11 import posixpath | |
12 import re | |
13 import six | |
14 import sys | |
15 | |
16 from errno import EINVAL, ENOENT, ENOTDIR, EBADF | |
17 from errno import EEXIST, EPERM, EACCES | |
18 from operator import attrgetter | |
19 from stat import ( | |
20 S_ISDIR, S_ISLNK, S_ISREG, S_ISSOCK, S_ISBLK, S_ISCHR, S_ISFIFO) | |
21 | |
22 try: | |
23 from collections.abc import Sequence | |
24 except ImportError: | |
25 from collections import Sequence | |
26 | |
27 try: | |
28 from urllib import quote as urlquote_from_bytes | |
29 except ImportError: | |
30 from urllib.parse import quote_from_bytes as urlquote_from_bytes | |
31 | |
32 | |
33 try: | |
34 intern = intern | |
35 except NameError: | |
36 intern = sys.intern | |
37 | |
38 supports_symlinks = True | |
39 if os.name == 'nt': | |
40 import nt | |
41 if sys.getwindowsversion()[:2] >= (6, 0) and sys.version_info >= (3, 2): | |
42 from nt import _getfinalpathname | |
43 else: | |
44 supports_symlinks = False | |
45 _getfinalpathname = None | |
46 else: | |
47 nt = None | |
48 | |
49 try: | |
50 from os import scandir as os_scandir | |
51 except ImportError: | |
52 from scandir import scandir as os_scandir | |
53 | |
54 __all__ = [ | |
55 "PurePath", "PurePosixPath", "PureWindowsPath", | |
56 "Path", "PosixPath", "WindowsPath", | |
57 ] | |
58 | |
59 # | |
60 # Internals | |
61 # | |
62 | |
63 # EBADF - guard agains macOS `stat` throwing EBADF | |
64 _IGNORED_ERROS = (ENOENT, ENOTDIR, EBADF) | |
65 | |
66 _IGNORED_WINERRORS = ( | |
67 21, # ERROR_NOT_READY - drive exists but is not accessible | |
68 ) | |
69 | |
70 | |
71 def _ignore_error(exception): | |
72 return (getattr(exception, 'errno', None) in _IGNORED_ERROS or | |
73 getattr(exception, 'winerror', None) in _IGNORED_WINERRORS) | |
74 | |
75 | |
76 def _py2_fsencode(parts): | |
77 # py2 => minimal unicode support | |
78 assert six.PY2 | |
79 return [part.encode('ascii') if isinstance(part, six.text_type) | |
80 else part for part in parts] | |
81 | |
82 | |
83 def _try_except_fileexistserror(try_func, except_func, else_func=None): | |
84 if sys.version_info >= (3, 3): | |
85 try: | |
86 try_func() | |
87 except FileExistsError as exc: | |
88 except_func(exc) | |
89 else: | |
90 if else_func is not None: | |
91 else_func() | |
92 else: | |
93 try: | |
94 try_func() | |
95 except EnvironmentError as exc: | |
96 if exc.errno != EEXIST: | |
97 raise | |
98 else: | |
99 except_func(exc) | |
100 else: | |
101 if else_func is not None: | |
102 else_func() | |
103 | |
104 | |
105 def _try_except_filenotfounderror(try_func, except_func): | |
106 if sys.version_info >= (3, 3): | |
107 try: | |
108 try_func() | |
109 except FileNotFoundError as exc: | |
110 except_func(exc) | |
111 elif os.name != 'nt': | |
112 try: | |
113 try_func() | |
114 except EnvironmentError as exc: | |
115 if exc.errno != ENOENT: | |
116 raise | |
117 else: | |
118 except_func(exc) | |
119 else: | |
120 try: | |
121 try_func() | |
122 except WindowsError as exc: | |
123 # errno contains winerror | |
124 # 2 = file not found | |
125 # 3 = path not found | |
126 if exc.errno not in (2, 3): | |
127 raise | |
128 else: | |
129 except_func(exc) | |
130 except EnvironmentError as exc: | |
131 if exc.errno != ENOENT: | |
132 raise | |
133 else: | |
134 except_func(exc) | |
135 | |
136 | |
137 def _try_except_permissionerror_iter(try_iter, except_iter): | |
138 if sys.version_info >= (3, 3): | |
139 try: | |
140 for x in try_iter(): | |
141 yield x | |
142 except PermissionError as exc: | |
143 for x in except_iter(exc): | |
144 yield x | |
145 else: | |
146 try: | |
147 for x in try_iter(): | |
148 yield x | |
149 except EnvironmentError as exc: | |
150 if exc.errno not in (EPERM, EACCES): | |
151 raise | |
152 else: | |
153 for x in except_iter(exc): | |
154 yield x | |
155 | |
156 | |
157 def _win32_get_unique_path_id(path): | |
158 # get file information, needed for samefile on older Python versions | |
159 # see http://timgolden.me.uk/python/win32_how_do_i/ | |
160 # see_if_two_files_are_the_same_file.html | |
161 from ctypes import POINTER, Structure, WinError | |
162 from ctypes.wintypes import DWORD, HANDLE, BOOL | |
163 | |
164 class FILETIME(Structure): | |
165 _fields_ = [("datetime_lo", DWORD), | |
166 ("datetime_hi", DWORD), | |
167 ] | |
168 | |
169 class BY_HANDLE_FILE_INFORMATION(Structure): | |
170 _fields_ = [("attributes", DWORD), | |
171 ("created_at", FILETIME), | |
172 ("accessed_at", FILETIME), | |
173 ("written_at", FILETIME), | |
174 ("volume", DWORD), | |
175 ("file_hi", DWORD), | |
176 ("file_lo", DWORD), | |
177 ("n_links", DWORD), | |
178 ("index_hi", DWORD), | |
179 ("index_lo", DWORD), | |
180 ] | |
181 | |
182 CreateFile = ctypes.windll.kernel32.CreateFileW | |
183 CreateFile.argtypes = [ctypes.c_wchar_p, DWORD, DWORD, ctypes.c_void_p, | |
184 DWORD, DWORD, HANDLE] | |
185 CreateFile.restype = HANDLE | |
186 GetFileInformationByHandle = ( | |
187 ctypes.windll.kernel32.GetFileInformationByHandle) | |
188 GetFileInformationByHandle.argtypes = [ | |
189 HANDLE, POINTER(BY_HANDLE_FILE_INFORMATION)] | |
190 GetFileInformationByHandle.restype = BOOL | |
191 CloseHandle = ctypes.windll.kernel32.CloseHandle | |
192 CloseHandle.argtypes = [HANDLE] | |
193 CloseHandle.restype = BOOL | |
194 GENERIC_READ = 0x80000000 | |
195 FILE_SHARE_READ = 0x00000001 | |
196 FILE_FLAG_BACKUP_SEMANTICS = 0x02000000 | |
197 OPEN_EXISTING = 3 | |
198 if os.path.isdir(path): | |
199 flags = FILE_FLAG_BACKUP_SEMANTICS | |
200 else: | |
201 flags = 0 | |
202 hfile = CreateFile(path, GENERIC_READ, FILE_SHARE_READ, | |
203 None, OPEN_EXISTING, flags, None) | |
204 if hfile == 0xffffffff: | |
205 if sys.version_info >= (3, 3): | |
206 raise FileNotFoundError(path) | |
207 else: | |
208 exc = OSError("file not found: path") | |
209 exc.errno = ENOENT | |
210 raise exc | |
211 info = BY_HANDLE_FILE_INFORMATION() | |
212 success = GetFileInformationByHandle(hfile, info) | |
213 CloseHandle(hfile) | |
214 if success == 0: | |
215 raise WinError() | |
216 return info.volume, info.index_hi, info.index_lo | |
217 | |
218 | |
219 def _is_wildcard_pattern(pat): | |
220 # Whether this pattern needs actual matching using fnmatch, or can | |
221 # be looked up directly as a file. | |
222 return "*" in pat or "?" in pat or "[" in pat | |
223 | |
224 | |
225 class _Flavour(object): | |
226 | |
227 """A flavour implements a particular (platform-specific) set of path | |
228 semantics.""" | |
229 | |
230 def __init__(self): | |
231 self.join = self.sep.join | |
232 | |
233 def parse_parts(self, parts): | |
234 if six.PY2: | |
235 parts = _py2_fsencode(parts) | |
236 parsed = [] | |
237 sep = self.sep | |
238 altsep = self.altsep | |
239 drv = root = '' | |
240 it = reversed(parts) | |
241 for part in it: | |
242 if not part: | |
243 continue | |
244 if altsep: | |
245 part = part.replace(altsep, sep) | |
246 drv, root, rel = self.splitroot(part) | |
247 if sep in rel: | |
248 for x in reversed(rel.split(sep)): | |
249 if x and x != '.': | |
250 parsed.append(intern(x)) | |
251 else: | |
252 if rel and rel != '.': | |
253 parsed.append(intern(rel)) | |
254 if drv or root: | |
255 if not drv: | |
256 # If no drive is present, try to find one in the previous | |
257 # parts. This makes the result of parsing e.g. | |
258 # ("C:", "/", "a") reasonably intuitive. | |
259 for part in it: | |
260 if not part: | |
261 continue | |
262 if altsep: | |
263 part = part.replace(altsep, sep) | |
264 drv = self.splitroot(part)[0] | |
265 if drv: | |
266 break | |
267 break | |
268 if drv or root: | |
269 parsed.append(drv + root) | |
270 parsed.reverse() | |
271 return drv, root, parsed | |
272 | |
273 def join_parsed_parts(self, drv, root, parts, drv2, root2, parts2): | |
274 """ | |
275 Join the two paths represented by the respective | |
276 (drive, root, parts) tuples. Return a new (drive, root, parts) tuple. | |
277 """ | |
278 if root2: | |
279 if not drv2 and drv: | |
280 return drv, root2, [drv + root2] + parts2[1:] | |
281 elif drv2: | |
282 if drv2 == drv or self.casefold(drv2) == self.casefold(drv): | |
283 # Same drive => second path is relative to the first | |
284 return drv, root, parts + parts2[1:] | |
285 else: | |
286 # Second path is non-anchored (common case) | |
287 return drv, root, parts + parts2 | |
288 return drv2, root2, parts2 | |
289 | |
290 | |
291 class _WindowsFlavour(_Flavour): | |
292 # Reference for Windows paths can be found at | |
293 # http://msdn.microsoft.com/en-us/library/aa365247%28v=vs.85%29.aspx | |
294 | |
295 sep = '\\' | |
296 altsep = '/' | |
297 has_drv = True | |
298 pathmod = ntpath | |
299 | |
300 is_supported = (os.name == 'nt') | |
301 | |
302 drive_letters = set('abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ') | |
303 ext_namespace_prefix = '\\\\?\\' | |
304 | |
305 reserved_names = ( | |
306 set(['CON', 'PRN', 'AUX', 'NUL']) | | |
307 set(['COM%d' % i for i in range(1, 10)]) | | |
308 set(['LPT%d' % i for i in range(1, 10)]) | |
309 ) | |
310 | |
311 # Interesting findings about extended paths: | |
312 # - '\\?\c:\a', '//?/c:\a' and '//?/c:/a' are all supported | |
313 # but '\\?\c:/a' is not | |
314 # - extended paths are always absolute; "relative" extended paths will | |
315 # fail. | |
316 | |
317 def splitroot(self, part, sep=sep): | |
318 first = part[0:1] | |
319 second = part[1:2] | |
320 if (second == sep and first == sep): | |
321 # XXX extended paths should also disable the collapsing of "." | |
322 # components (according to MSDN docs). | |
323 prefix, part = self._split_extended_path(part) | |
324 first = part[0:1] | |
325 second = part[1:2] | |
326 else: | |
327 prefix = '' | |
328 third = part[2:3] | |
329 if (second == sep and first == sep and third != sep): | |
330 # is a UNC path: | |
331 # vvvvvvvvvvvvvvvvvvvvv root | |
332 # \\machine\mountpoint\directory\etc\... | |
333 # directory ^^^^^^^^^^^^^^ | |
334 index = part.find(sep, 2) | |
335 if index != -1: | |
336 index2 = part.find(sep, index + 1) | |
337 # a UNC path can't have two slashes in a row | |
338 # (after the initial two) | |
339 if index2 != index + 1: | |
340 if index2 == -1: | |
341 index2 = len(part) | |
342 if prefix: | |
343 return prefix + part[1:index2], sep, part[index2 + 1:] | |
344 else: | |
345 return part[:index2], sep, part[index2 + 1:] | |
346 drv = root = '' | |
347 if second == ':' and first in self.drive_letters: | |
348 drv = part[:2] | |
349 part = part[2:] | |
350 first = third | |
351 if first == sep: | |
352 root = first | |
353 part = part.lstrip(sep) | |
354 return prefix + drv, root, part | |
355 | |
356 def casefold(self, s): | |
357 return s.lower() | |
358 | |
359 def casefold_parts(self, parts): | |
360 return [p.lower() for p in parts] | |
361 | |
362 def resolve(self, path, strict=False): | |
363 s = str(path) | |
364 if not s: | |
365 return os.getcwd() | |
366 previous_s = None | |
367 if _getfinalpathname is not None: | |
368 if strict: | |
369 return self._ext_to_normal(_getfinalpathname(s)) | |
370 else: | |
371 # End of the path after the first one not found | |
372 tail_parts = [] | |
373 | |
374 def _try_func(): | |
375 result[0] = self._ext_to_normal(_getfinalpathname(s)) | |
376 # if there was no exception, set flag to 0 | |
377 result[1] = 0 | |
378 | |
379 def _exc_func(exc): | |
380 pass | |
381 | |
382 while True: | |
383 result = [None, 1] | |
384 _try_except_filenotfounderror(_try_func, _exc_func) | |
385 if result[1] == 1: # file not found exception raised | |
386 previous_s = s | |
387 s, tail = os.path.split(s) | |
388 tail_parts.append(tail) | |
389 if previous_s == s: | |
390 return path | |
391 else: | |
392 s = result[0] | |
393 return os.path.join(s, *reversed(tail_parts)) | |
394 # Means fallback on absolute | |
395 return None | |
396 | |
397 def _split_extended_path(self, s, ext_prefix=ext_namespace_prefix): | |
398 prefix = '' | |
399 if s.startswith(ext_prefix): | |
400 prefix = s[:4] | |
401 s = s[4:] | |
402 if s.startswith('UNC\\'): | |
403 prefix += s[:3] | |
404 s = '\\' + s[3:] | |
405 return prefix, s | |
406 | |
407 def _ext_to_normal(self, s): | |
408 # Turn back an extended path into a normal DOS-like path | |
409 return self._split_extended_path(s)[1] | |
410 | |
411 def is_reserved(self, parts): | |
412 # NOTE: the rules for reserved names seem somewhat complicated | |
413 # (e.g. r"..\NUL" is reserved but not r"foo\NUL"). | |
414 # We err on the side of caution and return True for paths which are | |
415 # not considered reserved by Windows. | |
416 if not parts: | |
417 return False | |
418 if parts[0].startswith('\\\\'): | |
419 # UNC paths are never reserved | |
420 return False | |
421 return parts[-1].partition('.')[0].upper() in self.reserved_names | |
422 | |
423 def make_uri(self, path): | |
424 # Under Windows, file URIs use the UTF-8 encoding. | |
425 drive = path.drive | |
426 if len(drive) == 2 and drive[1] == ':': | |
427 # It's a path on a local drive => 'file:///c:/a/b' | |
428 rest = path.as_posix()[2:].lstrip('/') | |
429 return 'file:///%s/%s' % ( | |
430 drive, urlquote_from_bytes(rest.encode('utf-8'))) | |
431 else: | |
432 # It's a path on a network drive => 'file://host/share/a/b' | |
433 return 'file:' + urlquote_from_bytes( | |
434 path.as_posix().encode('utf-8')) | |
435 | |
436 def gethomedir(self, username): | |
437 if 'HOME' in os.environ: | |
438 userhome = os.environ['HOME'] | |
439 elif 'USERPROFILE' in os.environ: | |
440 userhome = os.environ['USERPROFILE'] | |
441 elif 'HOMEPATH' in os.environ: | |
442 try: | |
443 drv = os.environ['HOMEDRIVE'] | |
444 except KeyError: | |
445 drv = '' | |
446 userhome = drv + os.environ['HOMEPATH'] | |
447 else: | |
448 raise RuntimeError("Can't determine home directory") | |
449 | |
450 if username: | |
451 # Try to guess user home directory. By default all users | |
452 # directories are located in the same place and are named by | |
453 # corresponding usernames. If current user home directory points | |
454 # to nonstandard place, this guess is likely wrong. | |
455 if os.environ['USERNAME'] != username: | |
456 drv, root, parts = self.parse_parts((userhome,)) | |
457 if parts[-1] != os.environ['USERNAME']: | |
458 raise RuntimeError("Can't determine home directory " | |
459 "for %r" % username) | |
460 parts[-1] = username | |
461 if drv or root: | |
462 userhome = drv + root + self.join(parts[1:]) | |
463 else: | |
464 userhome = self.join(parts) | |
465 return userhome | |
466 | |
467 | |
468 class _PosixFlavour(_Flavour): | |
469 sep = '/' | |
470 altsep = '' | |
471 has_drv = False | |
472 pathmod = posixpath | |
473 | |
474 is_supported = (os.name != 'nt') | |
475 | |
476 def splitroot(self, part, sep=sep): | |
477 if part and part[0] == sep: | |
478 stripped_part = part.lstrip(sep) | |
479 # According to POSIX path resolution: | |
480 # http://pubs.opengroup.org/onlinepubs/009695399/basedefs/ | |
481 # xbd_chap04.html#tag_04_11 | |
482 # "A pathname that begins with two successive slashes may be | |
483 # interpreted in an implementation-defined manner, although more | |
484 # than two leading slashes shall be treated as a single slash". | |
485 if len(part) - len(stripped_part) == 2: | |
486 return '', sep * 2, stripped_part | |
487 else: | |
488 return '', sep, stripped_part | |
489 else: | |
490 return '', '', part | |
491 | |
492 def casefold(self, s): | |
493 return s | |
494 | |
495 def casefold_parts(self, parts): | |
496 return parts | |
497 | |
498 def resolve(self, path, strict=False): | |
499 sep = self.sep | |
500 accessor = path._accessor | |
501 seen = {} | |
502 | |
503 def _resolve(path, rest): | |
504 if rest.startswith(sep): | |
505 path = '' | |
506 | |
507 for name in rest.split(sep): | |
508 if not name or name == '.': | |
509 # current dir | |
510 continue | |
511 if name == '..': | |
512 # parent dir | |
513 path, _, _ = path.rpartition(sep) | |
514 continue | |
515 newpath = path + sep + name | |
516 if newpath in seen: | |
517 # Already seen this path | |
518 path = seen[newpath] | |
519 if path is not None: | |
520 # use cached value | |
521 continue | |
522 # The symlink is not resolved, so we must have a symlink | |
523 # loop. | |
524 raise RuntimeError("Symlink loop from %r" % newpath) | |
525 # Resolve the symbolic link | |
526 try: | |
527 target = accessor.readlink(newpath) | |
528 except OSError as e: | |
529 if e.errno != EINVAL and strict: | |
530 raise | |
531 # Not a symlink, or non-strict mode. We just leave the path | |
532 # untouched. | |
533 path = newpath | |
534 else: | |
535 seen[newpath] = None # not resolved symlink | |
536 path = _resolve(path, target) | |
537 seen[newpath] = path # resolved symlink | |
538 | |
539 return path | |
540 # NOTE: according to POSIX, getcwd() cannot contain path components | |
541 # which are symlinks. | |
542 base = '' if path.is_absolute() else os.getcwd() | |
543 return _resolve(base, str(path)) or sep | |
544 | |
545 def is_reserved(self, parts): | |
546 return False | |
547 | |
548 def make_uri(self, path): | |
549 # We represent the path using the local filesystem encoding, | |
550 # for portability to other applications. | |
551 bpath = bytes(path) | |
552 return 'file://' + urlquote_from_bytes(bpath) | |
553 | |
554 def gethomedir(self, username): | |
555 if not username: | |
556 try: | |
557 return os.environ['HOME'] | |
558 except KeyError: | |
559 import pwd | |
560 return pwd.getpwuid(os.getuid()).pw_dir | |
561 else: | |
562 import pwd | |
563 try: | |
564 return pwd.getpwnam(username).pw_dir | |
565 except KeyError: | |
566 raise RuntimeError("Can't determine home directory " | |
567 "for %r" % username) | |
568 | |
569 | |
570 _windows_flavour = _WindowsFlavour() | |
571 _posix_flavour = _PosixFlavour() | |
572 | |
573 | |
574 class _Accessor: | |
575 | |
576 """An accessor implements a particular (system-specific or not) way of | |
577 accessing paths on the filesystem.""" | |
578 | |
579 | |
580 class _NormalAccessor(_Accessor): | |
581 | |
582 def _wrap_strfunc(strfunc): | |
583 @functools.wraps(strfunc) | |
584 def wrapped(pathobj, *args): | |
585 return strfunc(str(pathobj), *args) | |
586 return staticmethod(wrapped) | |
587 | |
588 def _wrap_binary_strfunc(strfunc): | |
589 @functools.wraps(strfunc) | |
590 def wrapped(pathobjA, pathobjB, *args): | |
591 return strfunc(str(pathobjA), str(pathobjB), *args) | |
592 return staticmethod(wrapped) | |
593 | |
594 stat = _wrap_strfunc(os.stat) | |
595 | |
596 lstat = _wrap_strfunc(os.lstat) | |
597 | |
598 open = _wrap_strfunc(os.open) | |
599 | |
600 listdir = _wrap_strfunc(os.listdir) | |
601 | |
602 scandir = _wrap_strfunc(os_scandir) | |
603 | |
604 chmod = _wrap_strfunc(os.chmod) | |
605 | |
606 if hasattr(os, "lchmod"): | |
607 lchmod = _wrap_strfunc(os.lchmod) | |
608 else: | |
609 def lchmod(self, pathobj, mode): | |
610 raise NotImplementedError("lchmod() not available on this system") | |
611 | |
612 mkdir = _wrap_strfunc(os.mkdir) | |
613 | |
614 unlink = _wrap_strfunc(os.unlink) | |
615 | |
616 rmdir = _wrap_strfunc(os.rmdir) | |
617 | |
618 rename = _wrap_binary_strfunc(os.rename) | |
619 | |
620 if sys.version_info >= (3, 3): | |
621 replace = _wrap_binary_strfunc(os.replace) | |
622 | |
623 if nt: | |
624 if supports_symlinks: | |
625 symlink = _wrap_binary_strfunc(os.symlink) | |
626 else: | |
627 def symlink(a, b, target_is_directory): | |
628 raise NotImplementedError( | |
629 "symlink() not available on this system") | |
630 else: | |
631 # Under POSIX, os.symlink() takes two args | |
632 @staticmethod | |
633 def symlink(a, b, target_is_directory): | |
634 return os.symlink(str(a), str(b)) | |
635 | |
636 utime = _wrap_strfunc(os.utime) | |
637 | |
638 # Helper for resolve() | |
639 def readlink(self, path): | |
640 return os.readlink(path) | |
641 | |
642 | |
643 _normal_accessor = _NormalAccessor() | |
644 | |
645 | |
646 # | |
647 # Globbing helpers | |
648 # | |
649 | |
650 def _make_selector(pattern_parts): | |
651 pat = pattern_parts[0] | |
652 child_parts = pattern_parts[1:] | |
653 if pat == '**': | |
654 cls = _RecursiveWildcardSelector | |
655 elif '**' in pat: | |
656 raise ValueError( | |
657 "Invalid pattern: '**' can only be an entire path component") | |
658 elif _is_wildcard_pattern(pat): | |
659 cls = _WildcardSelector | |
660 else: | |
661 cls = _PreciseSelector | |
662 return cls(pat, child_parts) | |
663 | |
664 | |
665 if hasattr(functools, "lru_cache"): | |
666 _make_selector = functools.lru_cache()(_make_selector) | |
667 | |
668 | |
669 class _Selector: | |
670 | |
671 """A selector matches a specific glob pattern part against the children | |
672 of a given path.""" | |
673 | |
674 def __init__(self, child_parts): | |
675 self.child_parts = child_parts | |
676 if child_parts: | |
677 self.successor = _make_selector(child_parts) | |
678 self.dironly = True | |
679 else: | |
680 self.successor = _TerminatingSelector() | |
681 self.dironly = False | |
682 | |
683 def select_from(self, parent_path): | |
684 """Iterate over all child paths of `parent_path` matched by this | |
685 selector. This can contain parent_path itself.""" | |
686 path_cls = type(parent_path) | |
687 is_dir = path_cls.is_dir | |
688 exists = path_cls.exists | |
689 scandir = parent_path._accessor.scandir | |
690 if not is_dir(parent_path): | |
691 return iter([]) | |
692 return self._select_from(parent_path, is_dir, exists, scandir) | |
693 | |
694 | |
695 class _TerminatingSelector: | |
696 | |
697 def _select_from(self, parent_path, is_dir, exists, scandir): | |
698 yield parent_path | |
699 | |
700 | |
701 class _PreciseSelector(_Selector): | |
702 | |
703 def __init__(self, name, child_parts): | |
704 self.name = name | |
705 _Selector.__init__(self, child_parts) | |
706 | |
707 def _select_from(self, parent_path, is_dir, exists, scandir): | |
708 def try_iter(): | |
709 path = parent_path._make_child_relpath(self.name) | |
710 if (is_dir if self.dironly else exists)(path): | |
711 for p in self.successor._select_from( | |
712 path, is_dir, exists, scandir): | |
713 yield p | |
714 | |
715 def except_iter(exc): | |
716 return | |
717 yield | |
718 | |
719 for x in _try_except_permissionerror_iter(try_iter, except_iter): | |
720 yield x | |
721 | |
722 | |
723 class _WildcardSelector(_Selector): | |
724 | |
725 def __init__(self, pat, child_parts): | |
726 self.pat = re.compile(fnmatch.translate(pat)) | |
727 _Selector.__init__(self, child_parts) | |
728 | |
729 def _select_from(self, parent_path, is_dir, exists, scandir): | |
730 def try_iter(): | |
731 cf = parent_path._flavour.casefold | |
732 entries = list(scandir(parent_path)) | |
733 for entry in entries: | |
734 if not self.dironly or entry.is_dir(): | |
735 name = entry.name | |
736 casefolded = cf(name) | |
737 if self.pat.match(casefolded): | |
738 path = parent_path._make_child_relpath(name) | |
739 for p in self.successor._select_from( | |
740 path, is_dir, exists, scandir): | |
741 yield p | |
742 | |
743 def except_iter(exc): | |
744 return | |
745 yield | |
746 | |
747 for x in _try_except_permissionerror_iter(try_iter, except_iter): | |
748 yield x | |
749 | |
750 | |
751 class _RecursiveWildcardSelector(_Selector): | |
752 | |
753 def __init__(self, pat, child_parts): | |
754 _Selector.__init__(self, child_parts) | |
755 | |
756 def _iterate_directories(self, parent_path, is_dir, scandir): | |
757 yield parent_path | |
758 | |
759 def try_iter(): | |
760 entries = list(scandir(parent_path)) | |
761 for entry in entries: | |
762 entry_is_dir = False | |
763 try: | |
764 entry_is_dir = entry.is_dir() | |
765 except OSError as e: | |
766 if not _ignore_error(e): | |
767 raise | |
768 if entry_is_dir and not entry.is_symlink(): | |
769 path = parent_path._make_child_relpath(entry.name) | |
770 for p in self._iterate_directories(path, is_dir, scandir): | |
771 yield p | |
772 | |
773 def except_iter(exc): | |
774 return | |
775 yield | |
776 | |
777 for x in _try_except_permissionerror_iter(try_iter, except_iter): | |
778 yield x | |
779 | |
780 def _select_from(self, parent_path, is_dir, exists, scandir): | |
781 def try_iter(): | |
782 yielded = set() | |
783 try: | |
784 successor_select = self.successor._select_from | |
785 for starting_point in self._iterate_directories( | |
786 parent_path, is_dir, scandir): | |
787 for p in successor_select( | |
788 starting_point, is_dir, exists, scandir): | |
789 if p not in yielded: | |
790 yield p | |
791 yielded.add(p) | |
792 finally: | |
793 yielded.clear() | |
794 | |
795 def except_iter(exc): | |
796 return | |
797 yield | |
798 | |
799 for x in _try_except_permissionerror_iter(try_iter, except_iter): | |
800 yield x | |
801 | |
802 | |
803 # | |
804 # Public API | |
805 # | |
806 | |
807 class _PathParents(Sequence): | |
808 | |
809 """This object provides sequence-like access to the logical ancestors | |
810 of a path. Don't try to construct it yourself.""" | |
811 __slots__ = ('_pathcls', '_drv', '_root', '_parts') | |
812 | |
813 def __init__(self, path): | |
814 # We don't store the instance to avoid reference cycles | |
815 self._pathcls = type(path) | |
816 self._drv = path._drv | |
817 self._root = path._root | |
818 self._parts = path._parts | |
819 | |
820 def __len__(self): | |
821 if self._drv or self._root: | |
822 return len(self._parts) - 1 | |
823 else: | |
824 return len(self._parts) | |
825 | |
826 def __getitem__(self, idx): | |
827 if idx < 0 or idx >= len(self): | |
828 raise IndexError(idx) | |
829 return self._pathcls._from_parsed_parts(self._drv, self._root, | |
830 self._parts[:-idx - 1]) | |
831 | |
832 def __repr__(self): | |
833 return "<{0}.parents>".format(self._pathcls.__name__) | |
834 | |
835 | |
836 class PurePath(object): | |
837 | |
838 """PurePath represents a filesystem path and offers operations which | |
839 don't imply any actual filesystem I/O. Depending on your system, | |
840 instantiating a PurePath will return either a PurePosixPath or a | |
841 PureWindowsPath object. You can also instantiate either of these classes | |
842 directly, regardless of your system. | |
843 """ | |
844 __slots__ = ( | |
845 '_drv', '_root', '_parts', | |
846 '_str', '_hash', '_pparts', '_cached_cparts', | |
847 ) | |
848 | |
849 def __new__(cls, *args): | |
850 """Construct a PurePath from one or several strings and or existing | |
851 PurePath objects. The strings and path objects are combined so as | |
852 to yield a canonicalized path, which is incorporated into the | |
853 new PurePath object. | |
854 """ | |
855 if cls is PurePath: | |
856 cls = PureWindowsPath if os.name == 'nt' else PurePosixPath | |
857 return cls._from_parts(args) | |
858 | |
859 def __reduce__(self): | |
860 # Using the parts tuple helps share interned path parts | |
861 # when pickling related paths. | |
862 return (self.__class__, tuple(self._parts)) | |
863 | |
864 @classmethod | |
865 def _parse_args(cls, args): | |
866 # This is useful when you don't want to create an instance, just | |
867 # canonicalize some constructor arguments. | |
868 parts = [] | |
869 for a in args: | |
870 if isinstance(a, PurePath): | |
871 parts += a._parts | |
872 else: | |
873 if sys.version_info >= (3, 6): | |
874 a = os.fspath(a) | |
875 else: | |
876 # duck typing for older Python versions | |
877 if hasattr(a, "__fspath__"): | |
878 a = a.__fspath__() | |
879 if isinstance(a, str): | |
880 # Force-cast str subclasses to str (issue #21127) | |
881 parts.append(str(a)) | |
882 # also handle unicode for PY2 (six.text_type = unicode) | |
883 elif six.PY2 and isinstance(a, six.text_type): | |
884 # cast to str using filesystem encoding | |
885 # note: in rare circumstances, on Python < 3.2, | |
886 # getfilesystemencoding can return None, in that | |
887 # case fall back to ascii | |
888 parts.append(a.encode( | |
889 sys.getfilesystemencoding() or "ascii")) | |
890 else: | |
891 raise TypeError( | |
892 "argument should be a str object or an os.PathLike " | |
893 "object returning str, not %r" | |
894 % type(a)) | |
895 return cls._flavour.parse_parts(parts) | |
896 | |
897 @classmethod | |
898 def _from_parts(cls, args, init=True): | |
899 # We need to call _parse_args on the instance, so as to get the | |
900 # right flavour. | |
901 self = object.__new__(cls) | |
902 drv, root, parts = self._parse_args(args) | |
903 self._drv = drv | |
904 self._root = root | |
905 self._parts = parts | |
906 if init: | |
907 self._init() | |
908 return self | |
909 | |
910 @classmethod | |
911 def _from_parsed_parts(cls, drv, root, parts, init=True): | |
912 self = object.__new__(cls) | |
913 self._drv = drv | |
914 self._root = root | |
915 self._parts = parts | |
916 if init: | |
917 self._init() | |
918 return self | |
919 | |
920 @classmethod | |
921 def _format_parsed_parts(cls, drv, root, parts): | |
922 if drv or root: | |
923 return drv + root + cls._flavour.join(parts[1:]) | |
924 else: | |
925 return cls._flavour.join(parts) | |
926 | |
927 def _init(self): | |
928 # Overridden in concrete Path | |
929 pass | |
930 | |
931 def _make_child(self, args): | |
932 drv, root, parts = self._parse_args(args) | |
933 drv, root, parts = self._flavour.join_parsed_parts( | |
934 self._drv, self._root, self._parts, drv, root, parts) | |
935 return self._from_parsed_parts(drv, root, parts) | |
936 | |
937 def __str__(self): | |
938 """Return the string representation of the path, suitable for | |
939 passing to system calls.""" | |
940 try: | |
941 return self._str | |
942 except AttributeError: | |
943 self._str = self._format_parsed_parts(self._drv, self._root, | |
944 self._parts) or '.' | |
945 return self._str | |
946 | |
947 def __fspath__(self): | |
948 return str(self) | |
949 | |
950 def as_posix(self): | |
951 """Return the string representation of the path with forward (/) | |
952 slashes.""" | |
953 f = self._flavour | |
954 return str(self).replace(f.sep, '/') | |
955 | |
956 def __bytes__(self): | |
957 """Return the bytes representation of the path. This is only | |
958 recommended to use under Unix.""" | |
959 if sys.version_info < (3, 2): | |
960 raise NotImplementedError("needs Python 3.2 or later") | |
961 return os.fsencode(str(self)) | |
962 | |
963 def __repr__(self): | |
964 return "{0}({1!r})".format(self.__class__.__name__, self.as_posix()) | |
965 | |
966 def as_uri(self): | |
967 """Return the path as a 'file' URI.""" | |
968 if not self.is_absolute(): | |
969 raise ValueError("relative path can't be expressed as a file URI") | |
970 return self._flavour.make_uri(self) | |
971 | |
972 @property | |
973 def _cparts(self): | |
974 # Cached casefolded parts, for hashing and comparison | |
975 try: | |
976 return self._cached_cparts | |
977 except AttributeError: | |
978 self._cached_cparts = self._flavour.casefold_parts(self._parts) | |
979 return self._cached_cparts | |
980 | |
981 def __eq__(self, other): | |
982 if not isinstance(other, PurePath): | |
983 return NotImplemented | |
984 return ( | |
985 self._cparts == other._cparts | |
986 and self._flavour is other._flavour) | |
987 | |
988 def __ne__(self, other): | |
989 return not self == other | |
990 | |
991 def __hash__(self): | |
992 try: | |
993 return self._hash | |
994 except AttributeError: | |
995 self._hash = hash(tuple(self._cparts)) | |
996 return self._hash | |
997 | |
998 def __lt__(self, other): | |
999 if (not isinstance(other, PurePath) | |
1000 or self._flavour is not other._flavour): | |
1001 return NotImplemented | |
1002 return self._cparts < other._cparts | |
1003 | |
1004 def __le__(self, other): | |
1005 if (not isinstance(other, PurePath) | |
1006 or self._flavour is not other._flavour): | |
1007 return NotImplemented | |
1008 return self._cparts <= other._cparts | |
1009 | |
1010 def __gt__(self, other): | |
1011 if (not isinstance(other, PurePath) | |
1012 or self._flavour is not other._flavour): | |
1013 return NotImplemented | |
1014 return self._cparts > other._cparts | |
1015 | |
1016 def __ge__(self, other): | |
1017 if (not isinstance(other, PurePath) | |
1018 or self._flavour is not other._flavour): | |
1019 return NotImplemented | |
1020 return self._cparts >= other._cparts | |
1021 | |
1022 drive = property(attrgetter('_drv'), | |
1023 doc="""The drive prefix (letter or UNC path), if any.""") | |
1024 | |
1025 root = property(attrgetter('_root'), | |
1026 doc="""The root of the path, if any.""") | |
1027 | |
1028 @property | |
1029 def anchor(self): | |
1030 """The concatenation of the drive and root, or ''.""" | |
1031 anchor = self._drv + self._root | |
1032 return anchor | |
1033 | |
1034 @property | |
1035 def name(self): | |
1036 """The final path component, if any.""" | |
1037 parts = self._parts | |
1038 if len(parts) == (1 if (self._drv or self._root) else 0): | |
1039 return '' | |
1040 return parts[-1] | |
1041 | |
1042 @property | |
1043 def suffix(self): | |
1044 """The final component's last suffix, if any.""" | |
1045 name = self.name | |
1046 i = name.rfind('.') | |
1047 if 0 < i < len(name) - 1: | |
1048 return name[i:] | |
1049 else: | |
1050 return '' | |
1051 | |
1052 @property | |
1053 def suffixes(self): | |
1054 """A list of the final component's suffixes, if any.""" | |
1055 name = self.name | |
1056 if name.endswith('.'): | |
1057 return [] | |
1058 name = name.lstrip('.') | |
1059 return ['.' + suffix for suffix in name.split('.')[1:]] | |
1060 | |
1061 @property | |
1062 def stem(self): | |
1063 """The final path component, minus its last suffix.""" | |
1064 name = self.name | |
1065 i = name.rfind('.') | |
1066 if 0 < i < len(name) - 1: | |
1067 return name[:i] | |
1068 else: | |
1069 return name | |
1070 | |
1071 def with_name(self, name): | |
1072 """Return a new path with the file name changed.""" | |
1073 if not self.name: | |
1074 raise ValueError("%r has an empty name" % (self,)) | |
1075 drv, root, parts = self._flavour.parse_parts((name,)) | |
1076 if (not name or name[-1] in [self._flavour.sep, self._flavour.altsep] | |
1077 or drv or root or len(parts) != 1): | |
1078 raise ValueError("Invalid name %r" % (name)) | |
1079 return self._from_parsed_parts(self._drv, self._root, | |
1080 self._parts[:-1] + [name]) | |
1081 | |
1082 def with_suffix(self, suffix): | |
1083 """Return a new path with the file suffix changed. If the path | |
1084 has no suffix, add given suffix. If the given suffix is an empty | |
1085 string, remove the suffix from the path. | |
1086 """ | |
1087 # XXX if suffix is None, should the current suffix be removed? | |
1088 f = self._flavour | |
1089 if f.sep in suffix or f.altsep and f.altsep in suffix: | |
1090 raise ValueError("Invalid suffix %r" % (suffix)) | |
1091 if suffix and not suffix.startswith('.') or suffix == '.': | |
1092 raise ValueError("Invalid suffix %r" % (suffix)) | |
1093 name = self.name | |
1094 if not name: | |
1095 raise ValueError("%r has an empty name" % (self,)) | |
1096 old_suffix = self.suffix | |
1097 if not old_suffix: | |
1098 name = name + suffix | |
1099 else: | |
1100 name = name[:-len(old_suffix)] + suffix | |
1101 return self._from_parsed_parts(self._drv, self._root, | |
1102 self._parts[:-1] + [name]) | |
1103 | |
1104 def relative_to(self, *other): | |
1105 """Return the relative path to another path identified by the passed | |
1106 arguments. If the operation is not possible (because this is not | |
1107 a subpath of the other path), raise ValueError. | |
1108 """ | |
1109 # For the purpose of this method, drive and root are considered | |
1110 # separate parts, i.e.: | |
1111 # Path('c:/').relative_to('c:') gives Path('/') | |
1112 # Path('c:/').relative_to('/') raise ValueError | |
1113 if not other: | |
1114 raise TypeError("need at least one argument") | |
1115 parts = self._parts | |
1116 drv = self._drv | |
1117 root = self._root | |
1118 if root: | |
1119 abs_parts = [drv, root] + parts[1:] | |
1120 else: | |
1121 abs_parts = parts | |
1122 to_drv, to_root, to_parts = self._parse_args(other) | |
1123 if to_root: | |
1124 to_abs_parts = [to_drv, to_root] + to_parts[1:] | |
1125 else: | |
1126 to_abs_parts = to_parts | |
1127 n = len(to_abs_parts) | |
1128 cf = self._flavour.casefold_parts | |
1129 if (root or drv) if n == 0 else cf(abs_parts[:n]) != cf(to_abs_parts): | |
1130 formatted = self._format_parsed_parts(to_drv, to_root, to_parts) | |
1131 raise ValueError("{0!r} does not start with {1!r}" | |
1132 .format(str(self), str(formatted))) | |
1133 return self._from_parsed_parts('', root if n == 1 else '', | |
1134 abs_parts[n:]) | |
1135 | |
1136 @property | |
1137 def parts(self): | |
1138 """An object providing sequence-like access to the | |
1139 components in the filesystem path.""" | |
1140 # We cache the tuple to avoid building a new one each time .parts | |
1141 # is accessed. XXX is this necessary? | |
1142 try: | |
1143 return self._pparts | |
1144 except AttributeError: | |
1145 self._pparts = tuple(self._parts) | |
1146 return self._pparts | |
1147 | |
1148 def joinpath(self, *args): | |
1149 """Combine this path with one or several arguments, and return a | |
1150 new path representing either a subpath (if all arguments are relative | |
1151 paths) or a totally different path (if one of the arguments is | |
1152 anchored). | |
1153 """ | |
1154 return self._make_child(args) | |
1155 | |
1156 def __truediv__(self, key): | |
1157 return self._make_child((key,)) | |
1158 | |
1159 def __rtruediv__(self, key): | |
1160 return self._from_parts([key] + self._parts) | |
1161 | |
1162 if six.PY2: | |
1163 __div__ = __truediv__ | |
1164 __rdiv__ = __rtruediv__ | |
1165 | |
1166 @property | |
1167 def parent(self): | |
1168 """The logical parent of the path.""" | |
1169 drv = self._drv | |
1170 root = self._root | |
1171 parts = self._parts | |
1172 if len(parts) == 1 and (drv or root): | |
1173 return self | |
1174 return self._from_parsed_parts(drv, root, parts[:-1]) | |
1175 | |
1176 @property | |
1177 def parents(self): | |
1178 """A sequence of this path's logical parents.""" | |
1179 return _PathParents(self) | |
1180 | |
1181 def is_absolute(self): | |
1182 """True if the path is absolute (has both a root and, if applicable, | |
1183 a drive).""" | |
1184 if not self._root: | |
1185 return False | |
1186 return not self._flavour.has_drv or bool(self._drv) | |
1187 | |
1188 def is_reserved(self): | |
1189 """Return True if the path contains one of the special names reserved | |
1190 by the system, if any.""" | |
1191 return self._flavour.is_reserved(self._parts) | |
1192 | |
1193 def match(self, path_pattern): | |
1194 """ | |
1195 Return True if this path matches the given pattern. | |
1196 """ | |
1197 cf = self._flavour.casefold | |
1198 path_pattern = cf(path_pattern) | |
1199 drv, root, pat_parts = self._flavour.parse_parts((path_pattern,)) | |
1200 if not pat_parts: | |
1201 raise ValueError("empty pattern") | |
1202 if drv and drv != cf(self._drv): | |
1203 return False | |
1204 if root and root != cf(self._root): | |
1205 return False | |
1206 parts = self._cparts | |
1207 if drv or root: | |
1208 if len(pat_parts) != len(parts): | |
1209 return False | |
1210 pat_parts = pat_parts[1:] | |
1211 elif len(pat_parts) > len(parts): | |
1212 return False | |
1213 for part, pat in zip(reversed(parts), reversed(pat_parts)): | |
1214 if not fnmatch.fnmatchcase(part, pat): | |
1215 return False | |
1216 return True | |
1217 | |
1218 | |
1219 # Can't subclass os.PathLike from PurePath and keep the constructor | |
1220 # optimizations in PurePath._parse_args(). | |
1221 if sys.version_info >= (3, 6): | |
1222 os.PathLike.register(PurePath) | |
1223 | |
1224 | |
1225 class PurePosixPath(PurePath): | |
1226 _flavour = _posix_flavour | |
1227 __slots__ = () | |
1228 | |
1229 | |
1230 class PureWindowsPath(PurePath): | |
1231 """PurePath subclass for Windows systems. | |
1232 | |
1233 On a Windows system, instantiating a PurePath should return this object. | |
1234 However, you can also instantiate it directly on any system. | |
1235 """ | |
1236 _flavour = _windows_flavour | |
1237 __slots__ = () | |
1238 | |
1239 | |
1240 # Filesystem-accessing classes | |
1241 | |
1242 | |
1243 class Path(PurePath): | |
1244 """PurePath subclass that can make system calls. | |
1245 | |
1246 Path represents a filesystem path but unlike PurePath, also offers | |
1247 methods to do system calls on path objects. Depending on your system, | |
1248 instantiating a Path will return either a PosixPath or a WindowsPath | |
1249 object. You can also instantiate a PosixPath or WindowsPath directly, | |
1250 but cannot instantiate a WindowsPath on a POSIX system or vice versa. | |
1251 """ | |
1252 __slots__ = ( | |
1253 '_accessor', | |
1254 '_closed', | |
1255 ) | |
1256 | |
1257 def __new__(cls, *args, **kwargs): | |
1258 if cls is Path: | |
1259 cls = WindowsPath if os.name == 'nt' else PosixPath | |
1260 self = cls._from_parts(args, init=False) | |
1261 if not self._flavour.is_supported: | |
1262 raise NotImplementedError("cannot instantiate %r on your system" | |
1263 % (cls.__name__,)) | |
1264 self._init() | |
1265 return self | |
1266 | |
1267 def _init(self, | |
1268 # Private non-constructor arguments | |
1269 template=None, | |
1270 ): | |
1271 self._closed = False | |
1272 if template is not None: | |
1273 self._accessor = template._accessor | |
1274 else: | |
1275 self._accessor = _normal_accessor | |
1276 | |
1277 def _make_child_relpath(self, part): | |
1278 # This is an optimization used for dir walking. `part` must be | |
1279 # a single part relative to this path. | |
1280 parts = self._parts + [part] | |
1281 return self._from_parsed_parts(self._drv, self._root, parts) | |
1282 | |
1283 def __enter__(self): | |
1284 if self._closed: | |
1285 self._raise_closed() | |
1286 return self | |
1287 | |
1288 def __exit__(self, t, v, tb): | |
1289 self._closed = True | |
1290 | |
1291 def _raise_closed(self): | |
1292 raise ValueError("I/O operation on closed path") | |
1293 | |
1294 def _opener(self, name, flags, mode=0o666): | |
1295 # A stub for the opener argument to built-in open() | |
1296 return self._accessor.open(self, flags, mode) | |
1297 | |
1298 def _raw_open(self, flags, mode=0o777): | |
1299 """ | |
1300 Open the file pointed by this path and return a file descriptor, | |
1301 as os.open() does. | |
1302 """ | |
1303 if self._closed: | |
1304 self._raise_closed() | |
1305 return self._accessor.open(self, flags, mode) | |
1306 | |
1307 # Public API | |
1308 | |
1309 @classmethod | |
1310 def cwd(cls): | |
1311 """Return a new path pointing to the current working directory | |
1312 (as returned by os.getcwd()). | |
1313 """ | |
1314 return cls(os.getcwd()) | |
1315 | |
1316 @classmethod | |
1317 def home(cls): | |
1318 """Return a new path pointing to the user's home directory (as | |
1319 returned by os.path.expanduser('~')). | |
1320 """ | |
1321 return cls(cls()._flavour.gethomedir(None)) | |
1322 | |
1323 def samefile(self, other_path): | |
1324 """Return whether other_path is the same or not as this file | |
1325 (as returned by os.path.samefile()). | |
1326 """ | |
1327 if hasattr(os.path, "samestat"): | |
1328 st = self.stat() | |
1329 try: | |
1330 other_st = other_path.stat() | |
1331 except AttributeError: | |
1332 other_st = os.stat(other_path) | |
1333 return os.path.samestat(st, other_st) | |
1334 else: | |
1335 filename1 = six.text_type(self) | |
1336 filename2 = six.text_type(other_path) | |
1337 st1 = _win32_get_unique_path_id(filename1) | |
1338 st2 = _win32_get_unique_path_id(filename2) | |
1339 return st1 == st2 | |
1340 | |
1341 def iterdir(self): | |
1342 """Iterate over the files in this directory. Does not yield any | |
1343 result for the special paths '.' and '..'. | |
1344 """ | |
1345 if self._closed: | |
1346 self._raise_closed() | |
1347 for name in self._accessor.listdir(self): | |
1348 if name in ('.', '..'): | |
1349 # Yielding a path object for these makes little sense | |
1350 continue | |
1351 yield self._make_child_relpath(name) | |
1352 if self._closed: | |
1353 self._raise_closed() | |
1354 | |
1355 def glob(self, pattern): | |
1356 """Iterate over this subtree and yield all existing files (of any | |
1357 kind, including directories) matching the given relative pattern. | |
1358 """ | |
1359 if not pattern: | |
1360 raise ValueError("Unacceptable pattern: {0!r}".format(pattern)) | |
1361 pattern = self._flavour.casefold(pattern) | |
1362 drv, root, pattern_parts = self._flavour.parse_parts((pattern,)) | |
1363 if drv or root: | |
1364 raise NotImplementedError("Non-relative patterns are unsupported") | |
1365 selector = _make_selector(tuple(pattern_parts)) | |
1366 for p in selector.select_from(self): | |
1367 yield p | |
1368 | |
1369 def rglob(self, pattern): | |
1370 """Recursively yield all existing files (of any kind, including | |
1371 directories) matching the given relative pattern, anywhere in | |
1372 this subtree. | |
1373 """ | |
1374 pattern = self._flavour.casefold(pattern) | |
1375 drv, root, pattern_parts = self._flavour.parse_parts((pattern,)) | |
1376 if drv or root: | |
1377 raise NotImplementedError("Non-relative patterns are unsupported") | |
1378 selector = _make_selector(("**",) + tuple(pattern_parts)) | |
1379 for p in selector.select_from(self): | |
1380 yield p | |
1381 | |
1382 def absolute(self): | |
1383 """Return an absolute version of this path. This function works | |
1384 even if the path doesn't point to anything. | |
1385 | |
1386 No normalization is done, i.e. all '.' and '..' will be kept along. | |
1387 Use resolve() to get the canonical path to a file. | |
1388 """ | |
1389 # XXX untested yet! | |
1390 if self._closed: | |
1391 self._raise_closed() | |
1392 if self.is_absolute(): | |
1393 return self | |
1394 # FIXME this must defer to the specific flavour (and, under Windows, | |
1395 # use nt._getfullpathname()) | |
1396 obj = self._from_parts([os.getcwd()] + self._parts, init=False) | |
1397 obj._init(template=self) | |
1398 return obj | |
1399 | |
1400 def resolve(self, strict=False): | |
1401 """ | |
1402 Make the path absolute, resolving all symlinks on the way and also | |
1403 normalizing it (for example turning slashes into backslashes under | |
1404 Windows). | |
1405 """ | |
1406 if self._closed: | |
1407 self._raise_closed() | |
1408 s = self._flavour.resolve(self, strict=strict) | |
1409 if s is None: | |
1410 # No symlink resolution => for consistency, raise an error if | |
1411 # the path is forbidden | |
1412 # but not raise error if file does not exist (see issue #54). | |
1413 | |
1414 def _try_func(): | |
1415 self.stat() | |
1416 | |
1417 def _exc_func(exc): | |
1418 pass | |
1419 | |
1420 _try_except_filenotfounderror(_try_func, _exc_func) | |
1421 s = str(self.absolute()) | |
1422 else: | |
1423 # ensure s is a string (normpath requires this on older python) | |
1424 s = str(s) | |
1425 # Now we have no symlinks in the path, it's safe to normalize it. | |
1426 normed = self._flavour.pathmod.normpath(s) | |
1427 obj = self._from_parts((normed,), init=False) | |
1428 obj._init(template=self) | |
1429 return obj | |
1430 | |
1431 def stat(self): | |
1432 """ | |
1433 Return the result of the stat() system call on this path, like | |
1434 os.stat() does. | |
1435 """ | |
1436 return self._accessor.stat(self) | |
1437 | |
1438 def owner(self): | |
1439 """ | |
1440 Return the login name of the file owner. | |
1441 """ | |
1442 import pwd | |
1443 return pwd.getpwuid(self.stat().st_uid).pw_name | |
1444 | |
1445 def group(self): | |
1446 """ | |
1447 Return the group name of the file gid. | |
1448 """ | |
1449 import grp | |
1450 return grp.getgrgid(self.stat().st_gid).gr_name | |
1451 | |
1452 def open(self, mode='r', buffering=-1, encoding=None, | |
1453 errors=None, newline=None): | |
1454 """ | |
1455 Open the file pointed by this path and return a file object, as | |
1456 the built-in open() function does. | |
1457 """ | |
1458 if self._closed: | |
1459 self._raise_closed() | |
1460 if sys.version_info >= (3, 3): | |
1461 return io.open( | |
1462 str(self), mode, buffering, encoding, errors, newline, | |
1463 opener=self._opener) | |
1464 else: | |
1465 return io.open(str(self), mode, buffering, | |
1466 encoding, errors, newline) | |
1467 | |
1468 def read_bytes(self): | |
1469 """ | |
1470 Open the file in bytes mode, read it, and close the file. | |
1471 """ | |
1472 with self.open(mode='rb') as f: | |
1473 return f.read() | |
1474 | |
1475 def read_text(self, encoding=None, errors=None): | |
1476 """ | |
1477 Open the file in text mode, read it, and close the file. | |
1478 """ | |
1479 with self.open(mode='r', encoding=encoding, errors=errors) as f: | |
1480 return f.read() | |
1481 | |
1482 def write_bytes(self, data): | |
1483 """ | |
1484 Open the file in bytes mode, write to it, and close the file. | |
1485 """ | |
1486 if not isinstance(data, six.binary_type): | |
1487 raise TypeError( | |
1488 'data must be %s, not %s' % | |
1489 (six.binary_type.__name__, data.__class__.__name__)) | |
1490 with self.open(mode='wb') as f: | |
1491 return f.write(data) | |
1492 | |
1493 def write_text(self, data, encoding=None, errors=None): | |
1494 """ | |
1495 Open the file in text mode, write to it, and close the file. | |
1496 """ | |
1497 if not isinstance(data, six.text_type): | |
1498 raise TypeError( | |
1499 'data must be %s, not %s' % | |
1500 (six.text_type.__name__, data.__class__.__name__)) | |
1501 with self.open(mode='w', encoding=encoding, errors=errors) as f: | |
1502 return f.write(data) | |
1503 | |
1504 def touch(self, mode=0o666, exist_ok=True): | |
1505 """ | |
1506 Create this file with the given access mode, if it doesn't exist. | |
1507 """ | |
1508 if self._closed: | |
1509 self._raise_closed() | |
1510 if exist_ok: | |
1511 # First try to bump modification time | |
1512 # Implementation note: GNU touch uses the UTIME_NOW option of | |
1513 # the utimensat() / futimens() functions. | |
1514 try: | |
1515 self._accessor.utime(self, None) | |
1516 except OSError: | |
1517 # Avoid exception chaining | |
1518 pass | |
1519 else: | |
1520 return | |
1521 flags = os.O_CREAT | os.O_WRONLY | |
1522 if not exist_ok: | |
1523 flags |= os.O_EXCL | |
1524 fd = self._raw_open(flags, mode) | |
1525 os.close(fd) | |
1526 | |
1527 def mkdir(self, mode=0o777, parents=False, exist_ok=False): | |
1528 """ | |
1529 Create a new directory at this given path. | |
1530 """ | |
1531 if self._closed: | |
1532 self._raise_closed() | |
1533 | |
1534 def _try_func(): | |
1535 self._accessor.mkdir(self, mode) | |
1536 | |
1537 def _exc_func(exc): | |
1538 if not parents or self.parent == self: | |
1539 raise exc | |
1540 self.parent.mkdir(parents=True, exist_ok=True) | |
1541 self.mkdir(mode, parents=False, exist_ok=exist_ok) | |
1542 | |
1543 try: | |
1544 _try_except_filenotfounderror(_try_func, _exc_func) | |
1545 except OSError: | |
1546 # Cannot rely on checking for EEXIST, since the operating system | |
1547 # could give priority to other errors like EACCES or EROFS | |
1548 if not exist_ok or not self.is_dir(): | |
1549 raise | |
1550 | |
1551 def chmod(self, mode): | |
1552 """ | |
1553 Change the permissions of the path, like os.chmod(). | |
1554 """ | |
1555 if self._closed: | |
1556 self._raise_closed() | |
1557 self._accessor.chmod(self, mode) | |
1558 | |
1559 def lchmod(self, mode): | |
1560 """ | |
1561 Like chmod(), except if the path points to a symlink, the symlink's | |
1562 permissions are changed, rather than its target's. | |
1563 """ | |
1564 if self._closed: | |
1565 self._raise_closed() | |
1566 self._accessor.lchmod(self, mode) | |
1567 | |
1568 def unlink(self): | |
1569 """ | |
1570 Remove this file or link. | |
1571 If the path is a directory, use rmdir() instead. | |
1572 """ | |
1573 if self._closed: | |
1574 self._raise_closed() | |
1575 self._accessor.unlink(self) | |
1576 | |
1577 def rmdir(self): | |
1578 """ | |
1579 Remove this directory. The directory must be empty. | |
1580 """ | |
1581 if self._closed: | |
1582 self._raise_closed() | |
1583 self._accessor.rmdir(self) | |
1584 | |
1585 def lstat(self): | |
1586 """ | |
1587 Like stat(), except if the path points to a symlink, the symlink's | |
1588 status information is returned, rather than its target's. | |
1589 """ | |
1590 if self._closed: | |
1591 self._raise_closed() | |
1592 return self._accessor.lstat(self) | |
1593 | |
1594 def rename(self, target): | |
1595 """ | |
1596 Rename this path to the given path. | |
1597 """ | |
1598 if self._closed: | |
1599 self._raise_closed() | |
1600 self._accessor.rename(self, target) | |
1601 | |
1602 def replace(self, target): | |
1603 """ | |
1604 Rename this path to the given path, clobbering the existing | |
1605 destination if it exists. | |
1606 """ | |
1607 if sys.version_info < (3, 3): | |
1608 raise NotImplementedError("replace() is only available " | |
1609 "with Python 3.3 and later") | |
1610 if self._closed: | |
1611 self._raise_closed() | |
1612 self._accessor.replace(self, target) | |
1613 | |
1614 def symlink_to(self, target, target_is_directory=False): | |
1615 """ | |
1616 Make this path a symlink pointing to the given path. | |
1617 Note the order of arguments (self, target) is the reverse of | |
1618 os.symlink's. | |
1619 """ | |
1620 if self._closed: | |
1621 self._raise_closed() | |
1622 self._accessor.symlink(target, self, target_is_directory) | |
1623 | |
1624 # Convenience functions for querying the stat results | |
1625 | |
1626 def exists(self): | |
1627 """ | |
1628 Whether this path exists. | |
1629 """ | |
1630 try: | |
1631 self.stat() | |
1632 except OSError as e: | |
1633 if not _ignore_error(e): | |
1634 raise | |
1635 return False | |
1636 except ValueError: | |
1637 # Non-encodable path | |
1638 return False | |
1639 return True | |
1640 | |
1641 def is_dir(self): | |
1642 """ | |
1643 Whether this path is a directory. | |
1644 """ | |
1645 try: | |
1646 return S_ISDIR(self.stat().st_mode) | |
1647 except OSError as e: | |
1648 if not _ignore_error(e): | |
1649 raise | |
1650 # Path doesn't exist or is a broken symlink | |
1651 # (see https://bitbucket.org/pitrou/pathlib/issue/12/) | |
1652 return False | |
1653 except ValueError: | |
1654 # Non-encodable path | |
1655 return False | |
1656 | |
1657 def is_file(self): | |
1658 """ | |
1659 Whether this path is a regular file (also True for symlinks pointing | |
1660 to regular files). | |
1661 """ | |
1662 try: | |
1663 return S_ISREG(self.stat().st_mode) | |
1664 except OSError as e: | |
1665 if not _ignore_error(e): | |
1666 raise | |
1667 # Path doesn't exist or is a broken symlink | |
1668 # (see https://bitbucket.org/pitrou/pathlib/issue/12/) | |
1669 return False | |
1670 except ValueError: | |
1671 # Non-encodable path | |
1672 return False | |
1673 | |
1674 def is_mount(self): | |
1675 """ | |
1676 Check if this path is a POSIX mount point | |
1677 """ | |
1678 # Need to exist and be a dir | |
1679 if not self.exists() or not self.is_dir(): | |
1680 return False | |
1681 | |
1682 parent = Path(self.parent) | |
1683 try: | |
1684 parent_dev = parent.stat().st_dev | |
1685 except OSError: | |
1686 return False | |
1687 | |
1688 dev = self.stat().st_dev | |
1689 if dev != parent_dev: | |
1690 return True | |
1691 ino = self.stat().st_ino | |
1692 parent_ino = parent.stat().st_ino | |
1693 return ino == parent_ino | |
1694 | |
1695 def is_symlink(self): | |
1696 """ | |
1697 Whether this path is a symbolic link. | |
1698 """ | |
1699 try: | |
1700 return S_ISLNK(self.lstat().st_mode) | |
1701 except OSError as e: | |
1702 if not _ignore_error(e): | |
1703 raise | |
1704 # Path doesn't exist | |
1705 return False | |
1706 except ValueError: | |
1707 # Non-encodable path | |
1708 return False | |
1709 | |
1710 def is_block_device(self): | |
1711 """ | |
1712 Whether this path is a block device. | |
1713 """ | |
1714 try: | |
1715 return S_ISBLK(self.stat().st_mode) | |
1716 except OSError as e: | |
1717 if not _ignore_error(e): | |
1718 raise | |
1719 # Path doesn't exist or is a broken symlink | |
1720 # (see https://bitbucket.org/pitrou/pathlib/issue/12/) | |
1721 return False | |
1722 except ValueError: | |
1723 # Non-encodable path | |
1724 return False | |
1725 | |
1726 def is_char_device(self): | |
1727 """ | |
1728 Whether this path is a character device. | |
1729 """ | |
1730 try: | |
1731 return S_ISCHR(self.stat().st_mode) | |
1732 except OSError as e: | |
1733 if not _ignore_error(e): | |
1734 raise | |
1735 # Path doesn't exist or is a broken symlink | |
1736 # (see https://bitbucket.org/pitrou/pathlib/issue/12/) | |
1737 return False | |
1738 except ValueError: | |
1739 # Non-encodable path | |
1740 return False | |
1741 | |
1742 def is_fifo(self): | |
1743 """ | |
1744 Whether this path is a FIFO. | |
1745 """ | |
1746 try: | |
1747 return S_ISFIFO(self.stat().st_mode) | |
1748 except OSError as e: | |
1749 if not _ignore_error(e): | |
1750 raise | |
1751 # Path doesn't exist or is a broken symlink | |
1752 # (see https://bitbucket.org/pitrou/pathlib/issue/12/) | |
1753 return False | |
1754 except ValueError: | |
1755 # Non-encodable path | |
1756 return False | |
1757 | |
1758 def is_socket(self): | |
1759 """ | |
1760 Whether this path is a socket. | |
1761 """ | |
1762 try: | |
1763 return S_ISSOCK(self.stat().st_mode) | |
1764 except OSError as e: | |
1765 if not _ignore_error(e): | |
1766 raise | |
1767 # Path doesn't exist or is a broken symlink | |
1768 # (see https://bitbucket.org/pitrou/pathlib/issue/12/) | |
1769 return False | |
1770 except ValueError: | |
1771 # Non-encodable path | |
1772 return False | |
1773 | |
1774 def expanduser(self): | |
1775 """ Return a new path with expanded ~ and ~user constructs | |
1776 (as returned by os.path.expanduser) | |
1777 """ | |
1778 if (not (self._drv or self._root) | |
1779 and self._parts and self._parts[0][:1] == '~'): | |
1780 homedir = self._flavour.gethomedir(self._parts[0][1:]) | |
1781 return self._from_parts([homedir] + self._parts[1:]) | |
1782 | |
1783 return self | |
1784 | |
1785 | |
1786 class PosixPath(Path, PurePosixPath): | |
1787 """Path subclass for non-Windows systems. | |
1788 | |
1789 On a POSIX system, instantiating a Path should return this object. | |
1790 """ | |
1791 __slots__ = () | |
1792 | |
1793 | |
1794 class WindowsPath(Path, PureWindowsPath): | |
1795 """Path subclass for Windows systems. | |
1796 | |
1797 On a Windows system, instantiating a Path should return this object. | |
1798 """ | |
1799 __slots__ = () | |
1800 | |
1801 def owner(self): | |
1802 raise NotImplementedError("Path.owner() is unsupported on this system") | |
1803 | |
1804 def group(self): | |
1805 raise NotImplementedError("Path.group() is unsupported on this system") | |
1806 | |
1807 def is_mount(self): | |
1808 raise NotImplementedError( | |
1809 "Path.is_mount() is unsupported on this system") |