| 
									
										
										
										
											2022-07-07 12:59:29 -07:00
										 |  |  | """Object-oriented filesystem paths.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This module provides classes to represent abstract paths and concrete | 
					
						
							|  |  |  | paths with operations that have semantics appropriate for different | 
					
						
							|  |  |  | operating systems. | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | import fnmatch | 
					
						
							|  |  |  | import functools | 
					
						
							|  |  |  | import io | 
					
						
							|  |  |  | import ntpath | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | import posixpath | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | import sys | 
					
						
							| 
									
										
										
										
											2021-04-23 21:48:52 +01:00
										 |  |  | import warnings | 
					
						
							| 
									
										
										
										
											2017-09-26 00:55:55 +03:00
										 |  |  | from _collections_abc import Sequence | 
					
						
							| 
									
										
										
										
											2021-12-30 02:45:06 -05:00
										 |  |  | from errno import ENOENT, ENOTDIR, EBADF, ELOOP | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | from operator import attrgetter | 
					
						
							|  |  |  | from stat import S_ISDIR, S_ISLNK, S_ISREG, S_ISSOCK, S_ISBLK, S_ISCHR, S_ISFIFO | 
					
						
							| 
									
										
										
										
											2013-12-03 09:41:35 +01:00
										 |  |  | from urllib.parse import quote_from_bytes as urlquote_from_bytes | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | __all__ = [ | 
					
						
							|  |  |  |     "PurePath", "PurePosixPath", "PureWindowsPath", | 
					
						
							|  |  |  |     "Path", "PosixPath", "WindowsPath", | 
					
						
							|  |  |  |     ] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Internals | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  | # Reference for Windows paths can be found at | 
					
						
							|  |  |  | # https://learn.microsoft.com/en-gb/windows/win32/fileio/naming-a-file . | 
					
						
							|  |  |  | _WIN_RESERVED_NAMES = frozenset( | 
					
						
							|  |  |  |     {'CON', 'PRN', 'AUX', 'NUL', 'CONIN$', 'CONOUT$'} | | 
					
						
							|  |  |  |     {f'COM{c}' for c in '123456789\xb9\xb2\xb3'} | | 
					
						
							|  |  |  |     {f'LPT{c}' for c in '123456789\xb9\xb2\xb3'} | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-28 16:50:17 +01:00
										 |  |  | _WINERROR_NOT_READY = 21  # drive exists but is not accessible | 
					
						
							|  |  |  | _WINERROR_INVALID_NAME = 123  # fix for bpo-35306 | 
					
						
							|  |  |  | _WINERROR_CANT_RESOLVE_FILENAME = 1921  # broken symlink pointing to itself | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-06 14:57:17 -04:00
										 |  |  | # EBADF - guard against macOS `stat` throwing EBADF | 
					
						
							| 
									
										
										
										
											2021-12-30 02:45:06 -05:00
										 |  |  | _IGNORED_ERRNOS = (ENOENT, ENOTDIR, EBADF, ELOOP) | 
					
						
							| 
									
										
										
										
											2018-08-27 23:33:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  | _IGNORED_WINERRORS = ( | 
					
						
							| 
									
										
										
										
											2021-04-28 16:50:17 +01:00
										 |  |  |     _WINERROR_NOT_READY, | 
					
						
							|  |  |  |     _WINERROR_INVALID_NAME, | 
					
						
							|  |  |  |     _WINERROR_CANT_RESOLVE_FILENAME) | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | def _ignore_error(exception): | 
					
						
							| 
									
										
										
										
											2021-12-30 02:45:06 -05:00
										 |  |  |     return (getattr(exception, 'errno', None) in _IGNORED_ERRNOS or | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |             getattr(exception, 'winerror', None) in _IGNORED_WINERRORS) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | def _is_wildcard_pattern(pat): | 
					
						
							|  |  |  |     # Whether this pattern needs actual matching using fnmatch, or can | 
					
						
							|  |  |  |     # be looked up directly as a file. | 
					
						
							|  |  |  |     return "*" in pat or "?" in pat or "[" in pat | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Globbing helpers | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-03 20:14:12 +03:00
										 |  |  | @functools.lru_cache() | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  | def _make_selector(pattern_parts, flavour): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     pat = pattern_parts[0] | 
					
						
							|  |  |  |     child_parts = pattern_parts[1:] | 
					
						
							| 
									
										
										
										
											2022-05-11 07:14:25 +03:00
										 |  |  |     if not pat: | 
					
						
							|  |  |  |         return _TerminatingSelector() | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     if pat == '**': | 
					
						
							|  |  |  |         cls = _RecursiveWildcardSelector | 
					
						
							|  |  |  |     elif '**' in pat: | 
					
						
							|  |  |  |         raise ValueError("Invalid pattern: '**' can only be an entire path component") | 
					
						
							|  |  |  |     elif _is_wildcard_pattern(pat): | 
					
						
							|  |  |  |         cls = _WildcardSelector | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         cls = _PreciseSelector | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |     return cls(pat, child_parts, flavour) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _Selector: | 
					
						
							|  |  |  |     """A selector matches a specific glob pattern part against the children
 | 
					
						
							|  |  |  |     of a given path."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |     def __init__(self, child_parts, flavour): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         self.child_parts = child_parts | 
					
						
							|  |  |  |         if child_parts: | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |             self.successor = _make_selector(child_parts, flavour) | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |             self.dironly = True | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         else: | 
					
						
							|  |  |  |             self.successor = _TerminatingSelector() | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |             self.dironly = False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def select_from(self, parent_path): | 
					
						
							|  |  |  |         """Iterate over all child paths of `parent_path` matched by this
 | 
					
						
							|  |  |  |         selector.  This can contain parent_path itself."""
 | 
					
						
							|  |  |  |         path_cls = type(parent_path) | 
					
						
							|  |  |  |         is_dir = path_cls.is_dir | 
					
						
							|  |  |  |         exists = path_cls.exists | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         scandir = path_cls._scandir | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         normcase = path_cls._flavour.normcase | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |         if not is_dir(parent_path): | 
					
						
							|  |  |  |             return iter([]) | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return self._select_from(parent_path, is_dir, exists, scandir, normcase) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _TerminatingSelector: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     def _select_from(self, parent_path, is_dir, exists, scandir, normcase): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         yield parent_path | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _PreciseSelector(_Selector): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |     def __init__(self, name, child_parts, flavour): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         self.name = name | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |         _Selector.__init__(self, child_parts, flavour) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     def _select_from(self, parent_path, is_dir, exists, scandir, normcase): | 
					
						
							| 
									
										
										
										
											2016-01-06 09:42:07 -08:00
										 |  |  |         try: | 
					
						
							|  |  |  |             path = parent_path._make_child_relpath(self.name) | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |             if (is_dir if self.dironly else exists)(path): | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |                 for p in self.successor._select_from(path, is_dir, exists, scandir, normcase): | 
					
						
							| 
									
										
										
										
											2016-01-06 09:42:07 -08:00
										 |  |  |                     yield p | 
					
						
							|  |  |  |         except PermissionError: | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _WildcardSelector(_Selector): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |     def __init__(self, pat, child_parts, flavour): | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         self.match = re.compile(fnmatch.translate(flavour.normcase(pat))).fullmatch | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |         _Selector.__init__(self, child_parts, flavour) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     def _select_from(self, parent_path, is_dir, exists, scandir, normcase): | 
					
						
							| 
									
										
										
										
											2016-01-06 09:42:07 -08:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2022-07-20 22:34:13 +01:00
										 |  |  |             # We must close the scandir() object before proceeding to | 
					
						
							|  |  |  |             # avoid exhausting file descriptors when globbing deep trees. | 
					
						
							| 
									
										
										
										
											2020-03-11 18:42:03 +02:00
										 |  |  |             with scandir(parent_path) as scandir_it: | 
					
						
							|  |  |  |                 entries = list(scandir_it) | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |             for entry in entries: | 
					
						
							| 
									
										
										
										
											2020-03-07 17:53:20 +00:00
										 |  |  |                 if self.dironly: | 
					
						
							|  |  |  |                     try: | 
					
						
							|  |  |  |                         # "entry.is_dir()" can raise PermissionError | 
					
						
							|  |  |  |                         # in some cases (see bpo-38894), which is not | 
					
						
							|  |  |  |                         # among the errors ignored by _ignore_error() | 
					
						
							|  |  |  |                         if not entry.is_dir(): | 
					
						
							|  |  |  |                             continue | 
					
						
							|  |  |  |                     except OSError as e: | 
					
						
							|  |  |  |                         if not _ignore_error(e): | 
					
						
							|  |  |  |                             raise | 
					
						
							|  |  |  |                         continue | 
					
						
							|  |  |  |                 name = entry.name | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |                 if self.match(normcase(name)): | 
					
						
							| 
									
										
										
										
											2020-03-07 17:53:20 +00:00
										 |  |  |                     path = parent_path._make_child_relpath(name) | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |                     for p in self.successor._select_from(path, is_dir, exists, scandir, normcase): | 
					
						
							| 
									
										
										
										
											2020-03-07 17:53:20 +00:00
										 |  |  |                         yield p | 
					
						
							| 
									
										
										
										
											2016-01-06 09:42:07 -08:00
										 |  |  |         except PermissionError: | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return | 
					
						
							| 
									
										
										
										
											2016-01-06 09:42:07 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | class _RecursiveWildcardSelector(_Selector): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |     def __init__(self, pat, child_parts, flavour): | 
					
						
							|  |  |  |         _Selector.__init__(self, child_parts, flavour) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |     def _iterate_directories(self, parent_path, is_dir, scandir): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         yield parent_path | 
					
						
							| 
									
										
										
										
											2016-01-07 10:56:36 -08:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2022-07-20 22:34:13 +01:00
										 |  |  |             # We must close the scandir() object before proceeding to | 
					
						
							|  |  |  |             # avoid exhausting file descriptors when globbing deep trees. | 
					
						
							| 
									
										
										
										
											2020-03-11 18:42:03 +02:00
										 |  |  |             with scandir(parent_path) as scandir_it: | 
					
						
							|  |  |  |                 entries = list(scandir_it) | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |             for entry in entries: | 
					
						
							| 
									
										
										
										
											2018-08-27 23:33:45 +02:00
										 |  |  |                 entry_is_dir = False | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     entry_is_dir = entry.is_dir() | 
					
						
							|  |  |  |                 except OSError as e: | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |                     if not _ignore_error(e): | 
					
						
							| 
									
										
										
										
											2018-08-27 23:33:45 +02:00
										 |  |  |                         raise | 
					
						
							|  |  |  |                 if entry_is_dir and not entry.is_symlink(): | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |                     path = parent_path._make_child_relpath(entry.name) | 
					
						
							|  |  |  |                     for p in self._iterate_directories(path, is_dir, scandir): | 
					
						
							| 
									
										
										
										
											2016-01-07 10:56:36 -08:00
										 |  |  |                         yield p | 
					
						
							|  |  |  |         except PermissionError: | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     def _select_from(self, parent_path, is_dir, exists, scandir, normcase): | 
					
						
							| 
									
										
										
										
											2016-01-06 09:42:07 -08:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |             yielded = set() | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 successor_select = self.successor._select_from | 
					
						
							|  |  |  |                 for starting_point in self._iterate_directories(parent_path, is_dir, scandir): | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |                     for p in successor_select(starting_point, is_dir, exists, scandir, normcase): | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |                         if p not in yielded: | 
					
						
							|  |  |  |                             yield p | 
					
						
							|  |  |  |                             yielded.add(p) | 
					
						
							|  |  |  |             finally: | 
					
						
							|  |  |  |                 yielded.clear() | 
					
						
							| 
									
										
										
										
											2016-01-06 09:42:07 -08:00
										 |  |  |         except PermissionError: | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Public API | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _PathParents(Sequence): | 
					
						
							|  |  |  |     """This object provides sequence-like access to the logical ancestors
 | 
					
						
							|  |  |  |     of a path.  Don't try to construct it yourself.""" | 
					
						
							|  |  |  |     __slots__ = ('_pathcls', '_drv', '_root', '_parts') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, path): | 
					
						
							|  |  |  |         # We don't store the instance to avoid reference cycles | 
					
						
							|  |  |  |         self._pathcls = type(path) | 
					
						
							|  |  |  |         self._drv = path._drv | 
					
						
							|  |  |  |         self._root = path._root | 
					
						
							|  |  |  |         self._parts = path._parts | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __len__(self): | 
					
						
							|  |  |  |         if self._drv or self._root: | 
					
						
							|  |  |  |             return len(self._parts) - 1 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return len(self._parts) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __getitem__(self, idx): | 
					
						
							| 
									
										
										
										
											2020-11-20 09:40:39 -06:00
										 |  |  |         if isinstance(idx, slice): | 
					
						
							|  |  |  |             return tuple(self[i] for i in range(*idx.indices(len(self)))) | 
					
						
							| 
									
										
										
										
											2020-11-23 22:06:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if idx >= len(self) or idx < -len(self): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             raise IndexError(idx) | 
					
						
							| 
									
										
										
										
											2022-06-03 22:33:20 +01:00
										 |  |  |         if idx < 0: | 
					
						
							|  |  |  |             idx += len(self) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         return self._pathcls._from_parsed_parts(self._drv, self._root, | 
					
						
							|  |  |  |                                                 self._parts[:-idx - 1]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return "<{}.parents>".format(self._pathcls.__name__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class PurePath(object): | 
					
						
							| 
									
										
										
										
											2018-02-19 08:36:32 +09:00
										 |  |  |     """Base class for manipulating paths without I/O.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     PurePath represents a filesystem path and offers operations which | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     don't imply any actual filesystem I/O.  Depending on your system, | 
					
						
							|  |  |  |     instantiating a PurePath will return either a PurePosixPath or a | 
					
						
							|  |  |  |     PureWindowsPath object.  You can also instantiate either of these classes | 
					
						
							|  |  |  |     directly, regardless of your system. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     __slots__ = ( | 
					
						
							|  |  |  |         '_drv', '_root', '_parts', | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         '_str', '_hash', '_parts_tuple', '_parts_normcase_cached', | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     ) | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     _flavour = os.path | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __new__(cls, *args): | 
					
						
							|  |  |  |         """Construct a PurePath from one or several strings and or existing
 | 
					
						
							|  |  |  |         PurePath objects.  The strings and path objects are combined so as | 
					
						
							|  |  |  |         to yield a canonicalized path, which is incorporated into the | 
					
						
							|  |  |  |         new PurePath object. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if cls is PurePath: | 
					
						
							|  |  |  |             cls = PureWindowsPath if os.name == 'nt' else PurePosixPath | 
					
						
							|  |  |  |         return cls._from_parts(args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __reduce__(self): | 
					
						
							|  |  |  |         # Using the parts tuple helps share interned path parts | 
					
						
							|  |  |  |         # when pickling related paths. | 
					
						
							|  |  |  |         return (self.__class__, tuple(self._parts)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def _parse_parts(cls, parts): | 
					
						
							|  |  |  |         if not parts: | 
					
						
							|  |  |  |             return '', '', [] | 
					
						
							| 
									
										
										
										
											2023-03-05 22:00:56 +00:00
										 |  |  |         elif len(parts) == 1: | 
					
						
							|  |  |  |             path = os.fspath(parts[0]) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             path = cls._flavour.join(*parts) | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         sep = cls._flavour.sep | 
					
						
							|  |  |  |         altsep = cls._flavour.altsep | 
					
						
							| 
									
										
										
										
											2023-03-05 23:50:21 +00:00
										 |  |  |         if isinstance(path, str): | 
					
						
							|  |  |  |             # Force-cast str subclasses to str (issue #21127) | 
					
						
							|  |  |  |             path = str(path) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             raise TypeError( | 
					
						
							|  |  |  |                 "argument should be a str or an os.PathLike " | 
					
						
							|  |  |  |                 "object where __fspath__ returns a str, " | 
					
						
							|  |  |  |                 f"not {type(path).__name__!r}") | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         if altsep: | 
					
						
							|  |  |  |             path = path.replace(altsep, sep) | 
					
						
							| 
									
										
										
										
											2023-01-27 00:28:27 +00:00
										 |  |  |         drv, root, rel = cls._flavour.splitroot(path) | 
					
						
							|  |  |  |         if drv.startswith(sep): | 
					
						
							|  |  |  |             # pathlib assumes that UNC paths always have a root. | 
					
						
							|  |  |  |             root = sep | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         unfiltered_parsed = [drv + root] + rel.split(sep) | 
					
						
							|  |  |  |         parsed = [sys.intern(x) for x in unfiltered_parsed if x and x != '.'] | 
					
						
							|  |  |  |         return drv, root, parsed | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2021-04-07 01:26:37 +01:00
										 |  |  |     def _from_parts(cls, args): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         self = object.__new__(cls) | 
					
						
							| 
									
										
										
										
											2023-03-05 23:50:21 +00:00
										 |  |  |         drv, root, parts = self._parse_parts(args) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         self._drv = drv | 
					
						
							|  |  |  |         self._root = root | 
					
						
							|  |  |  |         self._parts = parts | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2021-04-07 01:26:37 +01:00
										 |  |  |     def _from_parsed_parts(cls, drv, root, parts): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         self = object.__new__(cls) | 
					
						
							|  |  |  |         self._drv = drv | 
					
						
							|  |  |  |         self._root = root | 
					
						
							|  |  |  |         self._parts = parts | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def _format_parsed_parts(cls, drv, root, parts): | 
					
						
							|  |  |  |         if drv or root: | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             return drv + root + cls._flavour.sep.join(parts[1:]) | 
					
						
							| 
									
										
										
										
											2023-03-10 17:29:04 +00:00
										 |  |  |         elif parts and cls._flavour.splitdrive(parts[0])[0]: | 
					
						
							|  |  |  |             parts = ['.'] + parts | 
					
						
							|  |  |  |         return cls._flavour.sep.join(parts) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __str__(self): | 
					
						
							|  |  |  |         """Return the string representation of the path, suitable for
 | 
					
						
							|  |  |  |         passing to system calls."""
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return self._str | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             self._str = self._format_parsed_parts(self._drv, self._root, | 
					
						
							|  |  |  |                                                   self._parts) or '.' | 
					
						
							|  |  |  |             return self._str | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-10 12:20:49 -07:00
										 |  |  |     def __fspath__(self): | 
					
						
							|  |  |  |         return str(self) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     def as_posix(self): | 
					
						
							|  |  |  |         """Return the string representation of the path with forward (/)
 | 
					
						
							|  |  |  |         slashes."""
 | 
					
						
							|  |  |  |         f = self._flavour | 
					
						
							|  |  |  |         return str(self).replace(f.sep, '/') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __bytes__(self): | 
					
						
							|  |  |  |         """Return the bytes representation of the path.  This is only
 | 
					
						
							|  |  |  |         recommended to use under Unix."""
 | 
					
						
							| 
									
										
										
										
											2017-03-25 13:42:11 +02:00
										 |  |  |         return os.fsencode(self) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         return "{}({!r})".format(self.__class__.__name__, self.as_posix()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def as_uri(self): | 
					
						
							|  |  |  |         """Return the path as a 'file' URI.""" | 
					
						
							|  |  |  |         if not self.is_absolute(): | 
					
						
							|  |  |  |             raise ValueError("relative path can't be expressed as a file URI") | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         drive = self._drv | 
					
						
							|  |  |  |         if len(drive) == 2 and drive[1] == ':': | 
					
						
							|  |  |  |             # It's a path on a local drive => 'file:///c:/a/b' | 
					
						
							|  |  |  |             prefix = 'file:///' + drive | 
					
						
							|  |  |  |             path = self.as_posix()[2:] | 
					
						
							|  |  |  |         elif drive: | 
					
						
							|  |  |  |             # It's a path on a network drive => 'file://host/share/a/b' | 
					
						
							|  |  |  |             prefix = 'file:' | 
					
						
							|  |  |  |             path = self.as_posix() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # It's a posix path => 'file:///etc/hosts' | 
					
						
							|  |  |  |             prefix = 'file://' | 
					
						
							|  |  |  |             path = str(self) | 
					
						
							|  |  |  |         return prefix + urlquote_from_bytes(os.fsencode(path)) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     def _parts_normcase(self): | 
					
						
							|  |  |  |         # Cached parts with normalized case, for hashing and comparison. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             return self._parts_normcase_cached | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         except AttributeError: | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             self._parts_normcase_cached = [self._flavour.normcase(p) for p in self._parts] | 
					
						
							|  |  |  |             return self._parts_normcase_cached | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __eq__(self, other): | 
					
						
							|  |  |  |         if not isinstance(other, PurePath): | 
					
						
							|  |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return self._parts_normcase == other._parts_normcase and self._flavour is other._flavour | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __hash__(self): | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return self._hash | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             self._hash = hash(tuple(self._parts_normcase)) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return self._hash | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __lt__(self, other): | 
					
						
							|  |  |  |         if not isinstance(other, PurePath) or self._flavour is not other._flavour: | 
					
						
							|  |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return self._parts_normcase < other._parts_normcase | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __le__(self, other): | 
					
						
							|  |  |  |         if not isinstance(other, PurePath) or self._flavour is not other._flavour: | 
					
						
							|  |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return self._parts_normcase <= other._parts_normcase | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __gt__(self, other): | 
					
						
							|  |  |  |         if not isinstance(other, PurePath) or self._flavour is not other._flavour: | 
					
						
							|  |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return self._parts_normcase > other._parts_normcase | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __ge__(self, other): | 
					
						
							|  |  |  |         if not isinstance(other, PurePath) or self._flavour is not other._flavour: | 
					
						
							|  |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return self._parts_normcase >= other._parts_normcase | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     drive = property(attrgetter('_drv'), | 
					
						
							|  |  |  |                      doc="""The drive prefix (letter or UNC path), if any.""") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     root = property(attrgetter('_root'), | 
					
						
							|  |  |  |                     doc="""The root of the path, if any.""") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def anchor(self): | 
					
						
							|  |  |  |         """The concatenation of the drive and root, or ''.""" | 
					
						
							|  |  |  |         anchor = self._drv + self._root | 
					
						
							|  |  |  |         return anchor | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def name(self): | 
					
						
							|  |  |  |         """The final path component, if any.""" | 
					
						
							|  |  |  |         parts = self._parts | 
					
						
							|  |  |  |         if len(parts) == (1 if (self._drv or self._root) else 0): | 
					
						
							|  |  |  |             return '' | 
					
						
							|  |  |  |         return parts[-1] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def suffix(self): | 
					
						
							| 
									
										
										
										
											2019-11-02 18:46:24 +02:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         The final component's last suffix, if any. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This includes the leading period. For example: '.txt' | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         name = self.name | 
					
						
							|  |  |  |         i = name.rfind('.') | 
					
						
							|  |  |  |         if 0 < i < len(name) - 1: | 
					
						
							|  |  |  |             return name[i:] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return '' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def suffixes(self): | 
					
						
							| 
									
										
										
										
											2019-11-02 18:46:24 +02:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         A list of the final component's suffixes, if any. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         These include the leading periods. For example: ['.tar', '.gz'] | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         name = self.name | 
					
						
							|  |  |  |         if name.endswith('.'): | 
					
						
							|  |  |  |             return [] | 
					
						
							|  |  |  |         name = name.lstrip('.') | 
					
						
							|  |  |  |         return ['.' + suffix for suffix in name.split('.')[1:]] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def stem(self): | 
					
						
							|  |  |  |         """The final path component, minus its last suffix.""" | 
					
						
							|  |  |  |         name = self.name | 
					
						
							|  |  |  |         i = name.rfind('.') | 
					
						
							|  |  |  |         if 0 < i < len(name) - 1: | 
					
						
							|  |  |  |             return name[:i] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def with_name(self, name): | 
					
						
							|  |  |  |         """Return a new path with the file name changed.""" | 
					
						
							|  |  |  |         if not self.name: | 
					
						
							|  |  |  |             raise ValueError("%r has an empty name" % (self,)) | 
					
						
							| 
									
										
										
										
											2023-01-27 00:28:27 +00:00
										 |  |  |         f = self._flavour | 
					
						
							|  |  |  |         drv, root, tail = f.splitroot(name) | 
					
						
							|  |  |  |         if drv or root or not tail or f.sep in tail or (f.altsep and f.altsep in tail): | 
					
						
							| 
									
										
										
										
											2014-07-06 21:31:12 -04:00
										 |  |  |             raise ValueError("Invalid name %r" % (name)) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         return self._from_parsed_parts(self._drv, self._root, | 
					
						
							|  |  |  |                                        self._parts[:-1] + [name]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 17:29:49 +02:00
										 |  |  |     def with_stem(self, stem): | 
					
						
							|  |  |  |         """Return a new path with the stem changed.""" | 
					
						
							|  |  |  |         return self.with_name(stem + self.suffix) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     def with_suffix(self, suffix): | 
					
						
							| 
									
										
										
										
											2018-08-03 22:49:42 +02:00
										 |  |  |         """Return a new path with the file suffix changed.  If the path
 | 
					
						
							|  |  |  |         has no suffix, add given suffix.  If the given suffix is an empty | 
					
						
							|  |  |  |         string, remove the suffix from the path. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2014-07-06 21:37:15 -04:00
										 |  |  |         f = self._flavour | 
					
						
							|  |  |  |         if f.sep in suffix or f.altsep and f.altsep in suffix: | 
					
						
							| 
									
										
										
										
											2018-08-11 08:45:06 +03:00
										 |  |  |             raise ValueError("Invalid suffix %r" % (suffix,)) | 
					
						
							| 
									
										
										
										
											2014-07-06 21:37:15 -04:00
										 |  |  |         if suffix and not suffix.startswith('.') or suffix == '.': | 
					
						
							| 
									
										
										
										
											2014-01-03 00:07:17 +01:00
										 |  |  |             raise ValueError("Invalid suffix %r" % (suffix)) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         name = self.name | 
					
						
							|  |  |  |         if not name: | 
					
						
							|  |  |  |             raise ValueError("%r has an empty name" % (self,)) | 
					
						
							|  |  |  |         old_suffix = self.suffix | 
					
						
							|  |  |  |         if not old_suffix: | 
					
						
							|  |  |  |             name = name + suffix | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             name = name[:-len(old_suffix)] + suffix | 
					
						
							|  |  |  |         return self._from_parsed_parts(self._drv, self._root, | 
					
						
							|  |  |  |                                        self._parts[:-1] + [name]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-17 00:14:27 +00:00
										 |  |  |     def relative_to(self, other, /, *_deprecated, walk_up=False): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """Return the relative path to another path identified by the passed
 | 
					
						
							|  |  |  |         arguments.  If the operation is not possible (because this is not | 
					
						
							| 
									
										
										
										
											2022-10-29 01:20:14 +02:00
										 |  |  |         related to the other path), raise ValueError. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The *walk_up* parameter controls whether `..` may be used to resolve | 
					
						
							|  |  |  |         the path. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-12-17 00:14:27 +00:00
										 |  |  |         if _deprecated: | 
					
						
							|  |  |  |             msg = ("support for supplying more than one positional argument " | 
					
						
							|  |  |  |                    "to pathlib.PurePath.relative_to() is deprecated and " | 
					
						
							|  |  |  |                    "scheduled for removal in Python {remove}") | 
					
						
							|  |  |  |             warnings._deprecated("pathlib.PurePath.relative_to(*args)", msg, | 
					
						
							|  |  |  |                                  remove=(3, 14)) | 
					
						
							| 
									
										
										
										
											2022-11-25 19:15:57 +00:00
										 |  |  |         path_cls = type(self) | 
					
						
							| 
									
										
										
										
											2022-12-17 00:14:27 +00:00
										 |  |  |         other = path_cls(other, *_deprecated) | 
					
						
							| 
									
										
										
										
											2022-11-25 19:15:57 +00:00
										 |  |  |         for step, path in enumerate([other] + list(other.parents)): | 
					
						
							|  |  |  |             if self.is_relative_to(path): | 
					
						
							| 
									
										
										
										
											2022-10-29 01:20:14 +02:00
										 |  |  |                 break | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2022-11-25 19:15:57 +00:00
										 |  |  |             raise ValueError(f"{str(self)!r} and {str(other)!r} have different anchors") | 
					
						
							|  |  |  |         if step and not walk_up: | 
					
						
							|  |  |  |             raise ValueError(f"{str(self)!r} is not in the subpath of {str(other)!r}") | 
					
						
							|  |  |  |         parts = ('..',) * step + self.parts[len(path.parts):] | 
					
						
							|  |  |  |         return path_cls(*parts) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-17 00:14:27 +00:00
										 |  |  |     def is_relative_to(self, other, /, *_deprecated): | 
					
						
							| 
									
										
										
										
											2019-08-13 14:54:02 -05:00
										 |  |  |         """Return True if the path is relative to another path or False.
 | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-12-17 00:14:27 +00:00
										 |  |  |         if _deprecated: | 
					
						
							|  |  |  |             msg = ("support for supplying more than one argument to " | 
					
						
							|  |  |  |                    "pathlib.PurePath.is_relative_to() is deprecated and " | 
					
						
							|  |  |  |                    "scheduled for removal in Python {remove}") | 
					
						
							|  |  |  |             warnings._deprecated("pathlib.PurePath.is_relative_to(*args)", | 
					
						
							|  |  |  |                                  msg, remove=(3, 14)) | 
					
						
							|  |  |  |         other = type(self)(other, *_deprecated) | 
					
						
							| 
									
										
										
										
											2022-11-25 19:15:57 +00:00
										 |  |  |         return other == self or other in self.parents | 
					
						
							| 
									
										
										
										
											2019-08-13 14:54:02 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def parts(self): | 
					
						
							|  |  |  |         """An object providing sequence-like access to the
 | 
					
						
							|  |  |  |         components in the filesystem path."""
 | 
					
						
							|  |  |  |         # We cache the tuple to avoid building a new one each time .parts | 
					
						
							|  |  |  |         # is accessed.  XXX is this necessary? | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             return self._parts_tuple | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         except AttributeError: | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             self._parts_tuple = tuple(self._parts) | 
					
						
							|  |  |  |             return self._parts_tuple | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def joinpath(self, *args): | 
					
						
							|  |  |  |         """Combine this path with one or several arguments, and return a
 | 
					
						
							|  |  |  |         new path representing either a subpath (if all arguments are relative | 
					
						
							|  |  |  |         paths) or a totally different path (if one of the arguments is | 
					
						
							|  |  |  |         anchored). | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         drv1, root1, parts1 = self._drv, self._root, self._parts | 
					
						
							| 
									
										
										
										
											2023-03-05 23:50:21 +00:00
										 |  |  |         drv2, root2, parts2 = self._parse_parts(args) | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         if root2: | 
					
						
							|  |  |  |             if not drv2 and drv1: | 
					
						
							|  |  |  |                 return self._from_parsed_parts(drv1, root2, [drv1 + root2] + parts2[1:]) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return self._from_parsed_parts(drv2, root2, parts2) | 
					
						
							|  |  |  |         elif drv2: | 
					
						
							|  |  |  |             if drv2 == drv1 or self._flavour.normcase(drv2) == self._flavour.normcase(drv1): | 
					
						
							|  |  |  |                 # Same drive => second path is relative to the first. | 
					
						
							|  |  |  |                 return self._from_parsed_parts(drv1, root1, parts1 + parts2[1:]) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return self._from_parsed_parts(drv2, root2, parts2) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # Second path is non-anchored (common case). | 
					
						
							|  |  |  |             return self._from_parsed_parts(drv1, root1, parts1 + parts2) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __truediv__(self, key): | 
					
						
							| 
									
										
										
										
											2019-08-08 01:41:10 -04:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             return self.joinpath(key) | 
					
						
							| 
									
										
										
										
											2019-08-08 01:41:10 -04:00
										 |  |  |         except TypeError: | 
					
						
							|  |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __rtruediv__(self, key): | 
					
						
							| 
									
										
										
										
											2019-08-08 01:41:10 -04:00
										 |  |  |         try: | 
					
						
							|  |  |  |             return self._from_parts([key] + self._parts) | 
					
						
							|  |  |  |         except TypeError: | 
					
						
							|  |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def parent(self): | 
					
						
							|  |  |  |         """The logical parent of the path.""" | 
					
						
							|  |  |  |         drv = self._drv | 
					
						
							|  |  |  |         root = self._root | 
					
						
							|  |  |  |         parts = self._parts | 
					
						
							|  |  |  |         if len(parts) == 1 and (drv or root): | 
					
						
							|  |  |  |             return self | 
					
						
							|  |  |  |         return self._from_parsed_parts(drv, root, parts[:-1]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def parents(self): | 
					
						
							|  |  |  |         """A sequence of this path's logical parents.""" | 
					
						
							|  |  |  |         return _PathParents(self) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_absolute(self): | 
					
						
							|  |  |  |         """True if the path is absolute (has both a root and, if applicable,
 | 
					
						
							|  |  |  |         a drive)."""
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         # ntpath.isabs() is defective - see GH-44626 . | 
					
						
							|  |  |  |         if self._flavour is ntpath: | 
					
						
							|  |  |  |             return bool(self._drv and self._root) | 
					
						
							|  |  |  |         return self._flavour.isabs(self) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def is_reserved(self): | 
					
						
							|  |  |  |         """Return True if the path contains one of the special names reserved
 | 
					
						
							|  |  |  |         by the system, if any."""
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         if self._flavour is posixpath or not self._parts: | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # NOTE: the rules for reserved names seem somewhat complicated | 
					
						
							|  |  |  |         # (e.g. r"..\NUL" is reserved but not r"foo\NUL" if "foo" does not | 
					
						
							|  |  |  |         # exist). We err on the side of caution and return True for paths | 
					
						
							|  |  |  |         # which are not considered reserved by Windows. | 
					
						
							|  |  |  |         if self._parts[0].startswith('\\\\'): | 
					
						
							|  |  |  |             # UNC paths are never reserved. | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         name = self._parts[-1].partition('.')[0].partition(':')[0].rstrip(' ') | 
					
						
							|  |  |  |         return name.upper() in _WIN_RESERVED_NAMES | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def match(self, path_pattern): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Return True if this path matches the given pattern. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         path_pattern = self._flavour.normcase(path_pattern) | 
					
						
							|  |  |  |         drv, root, pat_parts = self._parse_parts((path_pattern,)) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         if not pat_parts: | 
					
						
							|  |  |  |             raise ValueError("empty pattern") | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         parts = self._parts_normcase | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         if drv or root: | 
					
						
							|  |  |  |             if len(pat_parts) != len(parts): | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |         elif len(pat_parts) > len(parts): | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         for part, pat in zip(reversed(parts), reversed(pat_parts)): | 
					
						
							|  |  |  |             if not fnmatch.fnmatchcase(part, pat): | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-10 12:20:49 -07:00
										 |  |  | # Can't subclass os.PathLike from PurePath and keep the constructor | 
					
						
							| 
									
										
										
										
											2023-03-05 23:50:21 +00:00
										 |  |  | # optimizations in PurePath.__slots__. | 
					
						
							| 
									
										
										
										
											2016-06-10 12:20:49 -07:00
										 |  |  | os.PathLike.register(PurePath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | class PurePosixPath(PurePath): | 
					
						
							| 
									
										
										
										
											2018-02-19 08:36:32 +09:00
										 |  |  |     """PurePath subclass for non-Windows systems.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     On a POSIX system, instantiating a PurePath should return this object. | 
					
						
							|  |  |  |     However, you can also instantiate it directly on any system. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     _flavour = posixpath | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     __slots__ = () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class PureWindowsPath(PurePath): | 
					
						
							| 
									
										
										
										
											2018-02-19 08:36:32 +09:00
										 |  |  |     """PurePath subclass for Windows systems.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     On a Windows system, instantiating a PurePath should return this object. | 
					
						
							|  |  |  |     However, you can also instantiate it directly on any system. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     _flavour = ntpath | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     __slots__ = () | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Filesystem-accessing classes | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Path(PurePath): | 
					
						
							| 
									
										
										
										
											2018-02-19 08:36:32 +09:00
										 |  |  |     """PurePath subclass that can make system calls.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Path represents a filesystem path but unlike PurePath, also offers | 
					
						
							|  |  |  |     methods to do system calls on path objects. Depending on your system, | 
					
						
							|  |  |  |     instantiating a Path will return either a PosixPath or a WindowsPath | 
					
						
							|  |  |  |     object. You can also instantiate a PosixPath or WindowsPath directly, | 
					
						
							|  |  |  |     but cannot instantiate a WindowsPath on a POSIX system or vice versa. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2021-04-07 01:26:37 +01:00
										 |  |  |     __slots__ = () | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __new__(cls, *args, **kwargs): | 
					
						
							| 
									
										
										
										
											2023-01-14 02:05:43 +02:00
										 |  |  |         if kwargs: | 
					
						
							|  |  |  |             msg = ("support for supplying keyword arguments to pathlib.PurePath " | 
					
						
							|  |  |  |                    "is deprecated and scheduled for removal in Python {remove}") | 
					
						
							|  |  |  |             warnings._deprecated("pathlib.PurePath(**kwargs)", msg, remove=(3, 14)) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         if cls is Path: | 
					
						
							|  |  |  |             cls = WindowsPath if os.name == 'nt' else PosixPath | 
					
						
							| 
									
										
										
										
											2023-03-05 22:46:45 +00:00
										 |  |  |         return cls._from_parts(args) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def _make_child_relpath(self, part): | 
					
						
							|  |  |  |         # This is an optimization used for dir walking.  `part` must be | 
					
						
							|  |  |  |         # a single part relative to this path. | 
					
						
							|  |  |  |         parts = self._parts + [part] | 
					
						
							|  |  |  |         return self._from_parsed_parts(self._drv, self._root, parts) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							| 
									
										
										
										
											2022-02-08 21:01:37 +00:00
										 |  |  |         # In previous versions of pathlib, __exit__() marked this path as | 
					
						
							|  |  |  |         # closed; subsequent attempts to perform I/O would raise an IOError. | 
					
						
							|  |  |  |         # This functionality was never documented, and had the effect of | 
					
						
							|  |  |  |         # making Path objects mutable, contrary to PEP 428. | 
					
						
							|  |  |  |         # In Python 3.9 __exit__() was made a no-op. | 
					
						
							|  |  |  |         # In Python 3.11 __enter__() began emitting DeprecationWarning. | 
					
						
							|  |  |  |         # In Python 3.13 __enter__() and __exit__() should be removed. | 
					
						
							|  |  |  |         warnings.warn("pathlib.Path.__enter__() is deprecated and scheduled " | 
					
						
							|  |  |  |                       "for removal in Python 3.13; Path objects as a context " | 
					
						
							|  |  |  |                       "manager is a no-op", | 
					
						
							|  |  |  |                       DeprecationWarning, stacklevel=2) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __exit__(self, t, v, tb): | 
					
						
							| 
									
										
										
										
											2020-04-01 15:10:51 +01:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Public API | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def cwd(cls): | 
					
						
							| 
									
										
										
										
											2023-01-05 22:11:50 +00:00
										 |  |  |         """Return a new path pointing to the current working directory.""" | 
					
						
							|  |  |  |         # We call 'absolute()' rather than using 'os.getcwd()' directly to | 
					
						
							|  |  |  |         # enable users to replace the implementation of 'absolute()' in a | 
					
						
							|  |  |  |         # subclass and benefit from the new behaviour here. This works because | 
					
						
							|  |  |  |         # os.path.abspath('.') == os.getcwd(). | 
					
						
							|  |  |  |         return cls().absolute() | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-12 21:03:41 +01:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def home(cls): | 
					
						
							|  |  |  |         """Return a new path pointing to the user's home directory (as
 | 
					
						
							|  |  |  |         returned by os.path.expanduser('~')). | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-04-07 23:50:13 +01:00
										 |  |  |         return cls("~").expanduser() | 
					
						
							| 
									
										
										
										
											2015-01-12 21:03:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-13 10:50:15 +02:00
										 |  |  |     def samefile(self, other_path): | 
					
						
							| 
									
										
										
										
											2015-10-22 03:34:16 +03:00
										 |  |  |         """Return whether other_path is the same or not as this file
 | 
					
						
							| 
									
										
										
										
											2015-10-21 20:10:24 +03:00
										 |  |  |         (as returned by os.path.samefile()). | 
					
						
							| 
									
										
										
										
											2014-05-13 10:50:15 +02:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         st = self.stat() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             other_st = other_path.stat() | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |             other_st = self.__class__(other_path).stat() | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return self._flavour.samestat(st, other_st) | 
					
						
							| 
									
										
										
										
											2014-05-13 10:50:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     def iterdir(self): | 
					
						
							| 
									
										
										
										
											2022-11-10 01:05:07 +03:00
										 |  |  |         """Yield path objects of the directory contents.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The children are yielded in arbitrary order, and the | 
					
						
							|  |  |  |         special entries '.' and '..' are not included. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         for name in os.listdir(self): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             yield self._make_child_relpath(name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |     def _scandir(self): | 
					
						
							|  |  |  |         # bpo-24132: a future version of pathlib will support subclassing of | 
					
						
							|  |  |  |         # pathlib.Path to customize how the filesystem is accessed. This | 
					
						
							|  |  |  |         # includes scandir(), which is used to implement glob(). | 
					
						
							|  |  |  |         return os.scandir(self) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     def glob(self, pattern): | 
					
						
							|  |  |  |         """Iterate over this subtree and yield all existing files (of any
 | 
					
						
							| 
									
										
										
										
											2019-02-11 11:47:09 +01:00
										 |  |  |         kind, including directories) matching the given relative pattern. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2020-02-12 12:11:34 +02:00
										 |  |  |         sys.audit("pathlib.Path.glob", self, pattern) | 
					
						
							| 
									
										
										
										
											2016-01-30 17:50:48 +02:00
										 |  |  |         if not pattern: | 
					
						
							|  |  |  |             raise ValueError("Unacceptable pattern: {!r}".format(pattern)) | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         drv, root, pattern_parts = self._parse_parts((pattern,)) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         if drv or root: | 
					
						
							|  |  |  |             raise NotImplementedError("Non-relative patterns are unsupported") | 
					
						
							| 
									
										
										
										
											2022-05-11 07:14:25 +03:00
										 |  |  |         if pattern[-1] in (self._flavour.sep, self._flavour.altsep): | 
					
						
							|  |  |  |             pattern_parts.append('') | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |         selector = _make_selector(tuple(pattern_parts), self._flavour) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         for p in selector.select_from(self): | 
					
						
							|  |  |  |             yield p | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def rglob(self, pattern): | 
					
						
							|  |  |  |         """Recursively yield all existing files (of any kind, including
 | 
					
						
							| 
									
										
										
										
											2019-02-11 11:47:09 +01:00
										 |  |  |         directories) matching the given relative pattern, anywhere in | 
					
						
							|  |  |  |         this subtree. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2020-02-12 12:11:34 +02:00
										 |  |  |         sys.audit("pathlib.Path.rglob", self, pattern) | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         drv, root, pattern_parts = self._parse_parts((pattern,)) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         if drv or root: | 
					
						
							|  |  |  |             raise NotImplementedError("Non-relative patterns are unsupported") | 
					
						
							| 
									
										
										
										
											2022-05-11 07:43:04 +03:00
										 |  |  |         if pattern and pattern[-1] in (self._flavour.sep, self._flavour.altsep): | 
					
						
							| 
									
										
										
										
											2022-05-11 07:14:25 +03:00
										 |  |  |             pattern_parts.append('') | 
					
						
							| 
									
										
										
										
											2019-10-21 20:37:15 +03:00
										 |  |  |         selector = _make_selector(("**",) + tuple(pattern_parts), self._flavour) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         for p in selector.select_from(self): | 
					
						
							|  |  |  |             yield p | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def absolute(self): | 
					
						
							| 
									
										
										
										
											2022-01-28 23:40:55 +00:00
										 |  |  |         """Return an absolute version of this path by prepending the current
 | 
					
						
							|  |  |  |         working directory. No normalization or symlink resolution is performed. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         Use resolve() to get the canonical path to a file. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if self.is_absolute(): | 
					
						
							|  |  |  |             return self | 
					
						
							| 
									
										
										
										
											2023-02-17 14:08:14 +00:00
										 |  |  |         elif self._drv: | 
					
						
							|  |  |  |             # There is a CWD on each drive-letter drive. | 
					
						
							|  |  |  |             cwd = self._flavour.abspath(self._drv) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             cwd = os.getcwd() | 
					
						
							|  |  |  |         return self._from_parts([cwd] + self._parts) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-09 12:58:17 -08:00
										 |  |  |     def resolve(self, strict=False): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Make the path absolute, resolving all symlinks on the way and also | 
					
						
							| 
									
										
										
										
											2022-01-28 23:40:55 +00:00
										 |  |  |         normalizing it. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-04-28 16:50:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def check_eloop(e): | 
					
						
							|  |  |  |             winerror = getattr(e, 'winerror', 0) | 
					
						
							|  |  |  |             if e.errno == ELOOP or winerror == _WINERROR_CANT_RESOLVE_FILENAME: | 
					
						
							|  |  |  |                 raise RuntimeError("Symlink loop from %r" % e.filename) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             s = self._flavour.realpath(self, strict=strict) | 
					
						
							| 
									
										
										
										
											2021-04-28 16:50:17 +01:00
										 |  |  |         except OSError as e: | 
					
						
							|  |  |  |             check_eloop(e) | 
					
						
							|  |  |  |             raise | 
					
						
							|  |  |  |         p = self._from_parts((s,)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # In non-strict mode, realpath() doesn't raise on symlink loops. | 
					
						
							|  |  |  |         # Ensure we get an exception by calling stat() | 
					
						
							|  |  |  |         if not strict: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 p.stat() | 
					
						
							|  |  |  |             except OSError as e: | 
					
						
							|  |  |  |                 check_eloop(e) | 
					
						
							|  |  |  |         return p | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 16:53:39 +01:00
										 |  |  |     def stat(self, *, follow_symlinks=True): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Return the result of the stat() system call on this path, like | 
					
						
							|  |  |  |         os.stat() does. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         return os.stat(self, follow_symlinks=follow_symlinks) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def owner(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Return the login name of the file owner. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             import pwd | 
					
						
							|  |  |  |             return pwd.getpwuid(self.stat().st_uid).pw_name | 
					
						
							|  |  |  |         except ImportError: | 
					
						
							|  |  |  |             raise NotImplementedError("Path.owner() is unsupported on this system") | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def group(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Return the group name of the file gid. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             import grp | 
					
						
							|  |  |  |             return grp.getgrgid(self.stat().st_gid).gr_name | 
					
						
							|  |  |  |         except ImportError: | 
					
						
							|  |  |  |             raise NotImplementedError("Path.group() is unsupported on this system") | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def open(self, mode='r', buffering=-1, encoding=None, | 
					
						
							|  |  |  |              errors=None, newline=None): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Open the file pointed by this path and return a file object, as | 
					
						
							|  |  |  |         the built-in open() function does. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-03-29 12:28:14 +09:00
										 |  |  |         if "b" not in mode: | 
					
						
							|  |  |  |             encoding = io.text_encoding(encoding) | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         return io.open(self, mode, buffering, encoding, errors, newline) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-01 19:12:33 +02:00
										 |  |  |     def read_bytes(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Open the file in bytes mode, read it, and close the file. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         with self.open(mode='rb') as f: | 
					
						
							|  |  |  |             return f.read() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def read_text(self, encoding=None, errors=None): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Open the file in text mode, read it, and close the file. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-03-29 12:28:14 +09:00
										 |  |  |         encoding = io.text_encoding(encoding) | 
					
						
							| 
									
										
										
										
											2014-10-01 19:12:33 +02:00
										 |  |  |         with self.open(mode='r', encoding=encoding, errors=errors) as f: | 
					
						
							|  |  |  |             return f.read() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def write_bytes(self, data): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Open the file in bytes mode, write to it, and close the file. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         # type-check for the buffer interface before truncating the file | 
					
						
							|  |  |  |         view = memoryview(data) | 
					
						
							|  |  |  |         with self.open(mode='wb') as f: | 
					
						
							|  |  |  |             return f.write(view) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-21 05:08:19 +03:00
										 |  |  |     def write_text(self, data, encoding=None, errors=None, newline=None): | 
					
						
							| 
									
										
										
										
											2014-10-01 19:12:33 +02:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Open the file in text mode, write to it, and close the file. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if not isinstance(data, str): | 
					
						
							|  |  |  |             raise TypeError('data must be str, not %s' % | 
					
						
							|  |  |  |                             data.__class__.__name__) | 
					
						
							| 
									
										
										
										
											2021-03-29 12:28:14 +09:00
										 |  |  |         encoding = io.text_encoding(encoding) | 
					
						
							| 
									
										
										
										
											2020-10-21 05:08:19 +03:00
										 |  |  |         with self.open(mode='w', encoding=encoding, errors=errors, newline=newline) as f: | 
					
						
							| 
									
										
										
										
											2014-10-01 19:12:33 +02:00
										 |  |  |             return f.write(data) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-23 14:18:40 -07:00
										 |  |  |     def readlink(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Return the path to which the symbolic link points. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         if not hasattr(os, "readlink"): | 
					
						
							|  |  |  |             raise NotImplementedError("os.readlink() not available on this system") | 
					
						
							|  |  |  |         return self._from_parts((os.readlink(self),)) | 
					
						
							| 
									
										
										
										
											2019-10-23 14:18:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     def touch(self, mode=0o666, exist_ok=True): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Create this file with the given access mode, if it doesn't exist. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if exist_ok: | 
					
						
							|  |  |  |             # First try to bump modification time | 
					
						
							|  |  |  |             # Implementation note: GNU touch uses the UTIME_NOW option of | 
					
						
							|  |  |  |             # the utimensat() / futimens() functions. | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 os.utime(self, None) | 
					
						
							|  |  |  |             except OSError: | 
					
						
							|  |  |  |                 # Avoid exception chaining | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return | 
					
						
							|  |  |  |         flags = os.O_CREAT | os.O_WRONLY | 
					
						
							|  |  |  |         if not exist_ok: | 
					
						
							|  |  |  |             flags |= os.O_EXCL | 
					
						
							|  |  |  |         fd = os.open(self, flags, mode) | 
					
						
							|  |  |  |         os.close(fd) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-05 11:28:12 -04:00
										 |  |  |     def mkdir(self, mode=0o777, parents=False, exist_ok=False): | 
					
						
							| 
									
										
										
										
											2017-03-24 20:51:53 +02:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Create a new directory at this given path. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |             os.mkdir(self, mode) | 
					
						
							| 
									
										
										
										
											2017-03-24 20:51:53 +02:00
										 |  |  |         except FileNotFoundError: | 
					
						
							|  |  |  |             if not parents or self.parent == self: | 
					
						
							|  |  |  |                 raise | 
					
						
							| 
									
										
										
										
											2017-04-13 20:08:15 +02:00
										 |  |  |             self.parent.mkdir(parents=True, exist_ok=True) | 
					
						
							|  |  |  |             self.mkdir(mode, parents=False, exist_ok=exist_ok) | 
					
						
							| 
									
										
										
										
											2017-03-24 20:51:53 +02:00
										 |  |  |         except OSError: | 
					
						
							|  |  |  |             # Cannot rely on checking for EEXIST, since the operating system | 
					
						
							|  |  |  |             # could give priority to other errors like EACCES or EROFS | 
					
						
							|  |  |  |             if not exist_ok or not self.is_dir(): | 
					
						
							|  |  |  |                 raise | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 16:53:39 +01:00
										 |  |  |     def chmod(self, mode, *, follow_symlinks=True): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Change the permissions of the path, like os.chmod(). | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         os.chmod(self, mode, follow_symlinks=follow_symlinks) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def lchmod(self, mode): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Like chmod(), except if the path points to a symlink, the symlink's | 
					
						
							|  |  |  |         permissions are changed, rather than its target's. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-04-07 16:53:39 +01:00
										 |  |  |         self.chmod(mode, follow_symlinks=False) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-16 00:02:11 +02:00
										 |  |  |     def unlink(self, missing_ok=False): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         Remove this file or link. | 
					
						
							|  |  |  |         If the path is a directory, use rmdir() instead. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2019-05-16 00:02:11 +02:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |             os.unlink(self) | 
					
						
							| 
									
										
										
										
											2019-05-16 00:02:11 +02:00
										 |  |  |         except FileNotFoundError: | 
					
						
							|  |  |  |             if not missing_ok: | 
					
						
							|  |  |  |                 raise | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def rmdir(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Remove this directory.  The directory must be empty. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         os.rmdir(self) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def lstat(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Like stat(), except if the path points to a symlink, the symlink's | 
					
						
							|  |  |  |         status information is returned, rather than its target's. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-04-07 16:53:39 +01:00
										 |  |  |         return self.stat(follow_symlinks=False) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def rename(self, target): | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2020-10-03 12:52:13 +03:00
										 |  |  |         Rename this path to the target path. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The target path may be absolute or relative. Relative paths are | 
					
						
							|  |  |  |         interpreted relative to the current working directory, *not* the | 
					
						
							|  |  |  |         directory of the Path object. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Returns the new Path instance pointing to the target path. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         os.rename(self, target) | 
					
						
							| 
									
										
										
										
											2019-09-11 21:26:49 +08:00
										 |  |  |         return self.__class__(target) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def replace(self, target): | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2020-10-03 12:52:13 +03:00
										 |  |  |         Rename this path to the target path, overwriting if that path exists. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The target path may be absolute or relative. Relative paths are | 
					
						
							|  |  |  |         interpreted relative to the current working directory, *not* the | 
					
						
							|  |  |  |         directory of the Path object. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Returns the new Path instance pointing to the target path. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         os.replace(self, target) | 
					
						
							| 
									
										
										
										
											2019-09-11 21:26:49 +08:00
										 |  |  |         return self.__class__(target) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def symlink_to(self, target, target_is_directory=False): | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-04-07 16:56:32 +01:00
										 |  |  |         Make this path a symlink pointing to the target path. | 
					
						
							|  |  |  |         Note the order of arguments (link, target) is the reverse of os.symlink. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         if not hasattr(os, "symlink"): | 
					
						
							|  |  |  |             raise NotImplementedError("os.symlink() not available on this system") | 
					
						
							|  |  |  |         os.symlink(target, self, target_is_directory) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-23 21:48:52 +01:00
										 |  |  |     def hardlink_to(self, target): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Make this path a hard link pointing to the same file as *target*. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Note the order of arguments (self, target) is the reverse of os.link's. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         if not hasattr(os, "link"): | 
					
						
							|  |  |  |             raise NotImplementedError("os.link() not available on this system") | 
					
						
							|  |  |  |         os.link(target, self) | 
					
						
							| 
									
										
										
										
											2021-04-23 21:48:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-07 16:56:32 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     # Convenience functions for querying the stat results | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def exists(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path exists. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             self.stat() | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |             if not _ignore_error(e): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-18 11:28:51 +03:00
										 |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_dir(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a directory. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISDIR(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |             if not _ignore_error(e): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							| 
									
										
										
										
											2021-05-05 01:08:26 -07:00
										 |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-18 11:28:51 +03:00
										 |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def is_file(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a regular file (also True for symlinks pointing | 
					
						
							|  |  |  |         to regular files). | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISREG(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |             if not _ignore_error(e): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							| 
									
										
										
										
											2021-05-05 01:08:26 -07:00
										 |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-18 11:28:51 +03:00
										 |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-01 15:35:45 -07:00
										 |  |  |     def is_mount(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-08-05 23:37:44 +01:00
										 |  |  |         Check if this path is a mount point | 
					
						
							| 
									
										
										
										
											2017-08-01 15:35:45 -07:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return self._flavour.ismount(self) | 
					
						
							| 
									
										
										
										
											2017-08-01 15:35:45 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     def is_symlink(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a symbolic link. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISLNK(self.lstat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |             if not _ignore_error(e): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-18 11:28:51 +03:00
										 |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-22 09:19:34 -08:00
										 |  |  |     def is_junction(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a junction. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return self._flavour.isjunction(self) | 
					
						
							| 
									
										
										
										
											2022-11-22 09:19:34 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     def is_block_device(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a block device. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISBLK(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |             if not _ignore_error(e): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							| 
									
										
										
										
											2021-05-05 01:08:26 -07:00
										 |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-18 11:28:51 +03:00
										 |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def is_char_device(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a character device. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISCHR(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |             if not _ignore_error(e): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							| 
									
										
										
										
											2021-05-05 01:08:26 -07:00
										 |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-18 11:28:51 +03:00
										 |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def is_fifo(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a FIFO. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISFIFO(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |             if not _ignore_error(e): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							| 
									
										
										
										
											2021-05-05 01:08:26 -07:00
										 |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-18 11:28:51 +03:00
										 |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def is_socket(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a socket. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISSOCK(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							| 
									
										
										
										
											2019-02-03 23:08:18 -08:00
										 |  |  |             if not _ignore_error(e): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							| 
									
										
										
										
											2021-05-05 01:08:26 -07:00
										 |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return False | 
					
						
							| 
									
										
										
										
											2018-09-18 11:28:51 +03:00
										 |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-30 20:54:45 +01:00
										 |  |  |     def expanduser(self): | 
					
						
							|  |  |  |         """ Return a new path with expanded ~ and ~user constructs
 | 
					
						
							|  |  |  |         (as returned by os.path.expanduser) | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if (not (self._drv or self._root) and | 
					
						
							|  |  |  |             self._parts and self._parts[0][:1] == '~'): | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             homedir = self._flavour.expanduser(self._parts[0]) | 
					
						
							| 
									
										
										
										
											2021-04-07 23:50:13 +01:00
										 |  |  |             if homedir[:1] == "~": | 
					
						
							|  |  |  |                 raise RuntimeError("Could not determine home directory.") | 
					
						
							| 
									
										
										
										
											2023-03-10 17:29:04 +00:00
										 |  |  |             drv, root, parts = self._parse_parts((homedir,)) | 
					
						
							|  |  |  |             return self._from_parsed_parts(drv, root, parts + self._parts[1:]) | 
					
						
							| 
									
										
										
										
											2014-12-30 20:54:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-23 03:55:46 +04:00
										 |  |  |     def walk(self, top_down=True, on_error=None, follow_symlinks=False): | 
					
						
							|  |  |  |         """Walk the directory tree from this directory, similar to os.walk().""" | 
					
						
							|  |  |  |         sys.audit("pathlib.Path.walk", self, on_error, follow_symlinks) | 
					
						
							|  |  |  |         return self._walk(top_down, on_error, follow_symlinks) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _walk(self, top_down, on_error, follow_symlinks): | 
					
						
							|  |  |  |         # We may not have read permission for self, in which case we can't | 
					
						
							|  |  |  |         # get a list of the files the directory contains. os.walk | 
					
						
							|  |  |  |         # always suppressed the exception then, rather than blow up for a | 
					
						
							|  |  |  |         # minor reason when (say) a thousand readable directories are still | 
					
						
							|  |  |  |         # left to visit. That logic is copied here. | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             scandir_it = self._scandir() | 
					
						
							|  |  |  |         except OSError as error: | 
					
						
							|  |  |  |             if on_error is not None: | 
					
						
							|  |  |  |                 on_error(error) | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with scandir_it: | 
					
						
							|  |  |  |             dirnames = [] | 
					
						
							|  |  |  |             filenames = [] | 
					
						
							|  |  |  |             for entry in scandir_it: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     is_dir = entry.is_dir(follow_symlinks=follow_symlinks) | 
					
						
							|  |  |  |                 except OSError: | 
					
						
							|  |  |  |                     # Carried over from os.path.isdir(). | 
					
						
							|  |  |  |                     is_dir = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if is_dir: | 
					
						
							|  |  |  |                     dirnames.append(entry.name) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     filenames.append(entry.name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if top_down: | 
					
						
							|  |  |  |             yield self, dirnames, filenames | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for dirname in dirnames: | 
					
						
							|  |  |  |             dirpath = self._make_child_relpath(dirname) | 
					
						
							|  |  |  |             yield from dirpath._walk(top_down, on_error, follow_symlinks) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not top_down: | 
					
						
							|  |  |  |             yield self, dirnames, filenames | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | class PosixPath(Path, PurePosixPath): | 
					
						
							| 
									
										
										
										
											2018-02-19 08:36:32 +09:00
										 |  |  |     """Path subclass for non-Windows systems.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     On a POSIX system, instantiating a Path should return this object. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     __slots__ = () | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-05 22:46:45 +00:00
										 |  |  |     if os.name == 'nt': | 
					
						
							|  |  |  |         def __new__(cls, *args, **kwargs): | 
					
						
							|  |  |  |             raise NotImplementedError( | 
					
						
							|  |  |  |                 f"cannot instantiate {cls.__name__!r} on your system") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | class WindowsPath(Path, PureWindowsPath): | 
					
						
							| 
									
										
										
										
											2018-02-19 08:36:32 +09:00
										 |  |  |     """Path subclass for Windows systems.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     On a Windows system, instantiating a Path should return this object. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     __slots__ = () | 
					
						
							| 
									
										
										
										
											2023-03-05 22:46:45 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if os.name != 'nt': | 
					
						
							|  |  |  |         def __new__(cls, *args, **kwargs): | 
					
						
							|  |  |  |             raise NotImplementedError( | 
					
						
							|  |  |  |                 f"cannot instantiate {cls.__name__!r} on your system") |