| 
									
										
										
										
											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 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) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-02 22:51:18 +01:00
										 |  |  | def _is_case_sensitive(flavour): | 
					
						
							|  |  |  |     return flavour.normcase('Aa') == 'Aa' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | # | 
					
						
							|  |  |  | # Globbing helpers | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-03 20:14:12 +03:00
										 |  |  | @functools.lru_cache() | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  | def _make_selector(pattern_parts, flavour, case_sensitive): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     pat = pattern_parts[0] | 
					
						
							| 
									
										
										
										
											2022-05-11 07:14:25 +03:00
										 |  |  |     if not pat: | 
					
						
							|  |  |  |         return _TerminatingSelector() | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     if pat == '**': | 
					
						
							| 
									
										
										
										
											2023-05-07 22:12:50 +01:00
										 |  |  |         child_parts_idx = 1 | 
					
						
							|  |  |  |         while child_parts_idx < len(pattern_parts) and pattern_parts[child_parts_idx] == '**': | 
					
						
							|  |  |  |             child_parts_idx += 1 | 
					
						
							|  |  |  |         child_parts = pattern_parts[child_parts_idx:] | 
					
						
							|  |  |  |         if '**' in child_parts: | 
					
						
							|  |  |  |             cls = _DoubleRecursiveWildcardSelector | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             cls = _RecursiveWildcardSelector | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2023-05-07 22:12:50 +01:00
										 |  |  |         child_parts = pattern_parts[1:] | 
					
						
							|  |  |  |         if pat == '..': | 
					
						
							|  |  |  |             cls = _ParentSelector | 
					
						
							|  |  |  |         elif '**' in pat: | 
					
						
							|  |  |  |             raise ValueError("Invalid pattern: '**' can only be an entire path component") | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             cls = _WildcardSelector | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  |     return cls(pat, child_parts, flavour, case_sensitive) | 
					
						
							| 
									
										
										
										
											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."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  |     def __init__(self, child_parts, flavour, case_sensitive): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         self.child_parts = child_parts | 
					
						
							|  |  |  |         if child_parts: | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  |             self.successor = _make_selector(child_parts, flavour, case_sensitive) | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							| 
									
										
										
										
											2022-02-02 12:38:25 +00:00
										 |  |  |         scandir = path_cls._scandir | 
					
						
							| 
									
										
										
										
											2023-05-03 17:28:44 +01:00
										 |  |  |         if not parent_path.is_dir(): | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |             return iter([]) | 
					
						
							| 
									
										
										
										
											2023-05-03 17:28:44 +01:00
										 |  |  |         return self._select_from(parent_path, scandir) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _TerminatingSelector: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-03 17:28:44 +01:00
										 |  |  |     def _select_from(self, parent_path, scandir): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         yield parent_path | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-03 00:16:04 +01:00
										 |  |  | class _ParentSelector(_Selector): | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name, child_parts, flavour, case_sensitive): | 
					
						
							|  |  |  |         _Selector.__init__(self, child_parts, flavour, case_sensitive) | 
					
						
							| 
									
										
										
										
											2023-05-03 00:16:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-03 17:28:44 +01:00
										 |  |  |     def _select_from(self,  parent_path, scandir): | 
					
						
							| 
									
										
										
										
											2023-05-03 00:16:04 +01:00
										 |  |  |         path = parent_path._make_child_relpath('..') | 
					
						
							| 
									
										
										
										
											2023-05-03 17:28:44 +01:00
										 |  |  |         for p in self.successor._select_from(path, scandir): | 
					
						
							| 
									
										
										
										
											2023-05-03 00:16:04 +01:00
										 |  |  |             yield p | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | class _WildcardSelector(_Selector): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  |     def __init__(self, pat, child_parts, flavour, case_sensitive): | 
					
						
							|  |  |  |         _Selector.__init__(self, child_parts, flavour, case_sensitive) | 
					
						
							|  |  |  |         if case_sensitive is None: | 
					
						
							|  |  |  |             # TODO: evaluate case-sensitivity of each directory in _select_from() | 
					
						
							|  |  |  |             case_sensitive = _is_case_sensitive(flavour) | 
					
						
							|  |  |  |         flags = re.NOFLAG if case_sensitive else re.IGNORECASE | 
					
						
							| 
									
										
										
										
											2023-05-02 22:51:18 +01:00
										 |  |  |         self.match = re.compile(fnmatch.translate(pat), flags=flags).fullmatch | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-03 17:28:44 +01:00
										 |  |  |     def _select_from(self, parent_path, scandir): | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							| 
									
										
										
										
											2023-05-11 01:01:39 +01:00
										 |  |  |         except OSError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2016-09-07 10:58:05 +03:00
										 |  |  |             for entry in entries: | 
					
						
							| 
									
										
										
										
											2020-03-07 17:53:20 +00:00
										 |  |  |                 if self.dironly: | 
					
						
							|  |  |  |                     try: | 
					
						
							|  |  |  |                         if not entry.is_dir(): | 
					
						
							|  |  |  |                             continue | 
					
						
							| 
									
										
										
										
											2023-05-11 01:01:39 +01:00
										 |  |  |                     except OSError: | 
					
						
							| 
									
										
										
										
											2020-03-07 17:53:20 +00:00
										 |  |  |                         continue | 
					
						
							|  |  |  |                 name = entry.name | 
					
						
							| 
									
										
										
										
											2023-05-02 22:51:18 +01:00
										 |  |  |                 if self.match(name): | 
					
						
							| 
									
										
										
										
											2020-03-07 17:53:20 +00:00
										 |  |  |                     path = parent_path._make_child_relpath(name) | 
					
						
							| 
									
										
										
										
											2023-05-03 17:28:44 +01:00
										 |  |  |                     for p in self.successor._select_from(path, scandir): | 
					
						
							| 
									
										
										
										
											2020-03-07 17:53:20 +00:00
										 |  |  |                         yield p | 
					
						
							| 
									
										
										
										
											2016-01-06 09:42:07 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | class _RecursiveWildcardSelector(_Selector): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  |     def __init__(self, pat, child_parts, flavour, case_sensitive): | 
					
						
							|  |  |  |         _Selector.__init__(self, child_parts, flavour, case_sensitive) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-15 18:33:32 +01:00
										 |  |  |     def _iterate_directories(self, parent_path): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         yield parent_path | 
					
						
							| 
									
										
										
										
											2023-05-15 18:33:32 +01:00
										 |  |  |         for dirpath, dirnames, _ in parent_path.walk(): | 
					
						
							|  |  |  |             for dirname in dirnames: | 
					
						
							|  |  |  |                 yield dirpath._make_child_relpath(dirname) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-03 17:28:44 +01:00
										 |  |  |     def _select_from(self, parent_path, scandir): | 
					
						
							| 
									
										
										
										
											2023-05-11 01:01:39 +01:00
										 |  |  |         successor_select = self.successor._select_from | 
					
						
							| 
									
										
										
										
											2023-05-15 18:33:32 +01:00
										 |  |  |         for starting_point in self._iterate_directories(parent_path): | 
					
						
							| 
									
										
										
										
											2023-05-11 01:01:39 +01:00
										 |  |  |             for p in successor_select(starting_point, scandir): | 
					
						
							|  |  |  |                 yield p | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 22:12:50 +01:00
										 |  |  | class _DoubleRecursiveWildcardSelector(_RecursiveWildcardSelector): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     Like _RecursiveWildcardSelector, but also de-duplicates results from | 
					
						
							|  |  |  |     successive selectors. This is necessary if the pattern contains | 
					
						
							|  |  |  |     multiple non-adjacent '**' segments. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _select_from(self, parent_path, scandir): | 
					
						
							|  |  |  |         yielded = set() | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             for p in super()._select_from(parent_path, scandir): | 
					
						
							|  |  |  |                 if p not in yielded: | 
					
						
							|  |  |  |                     yield p | 
					
						
							|  |  |  |                     yielded.add(p) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             yielded.clear() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | # | 
					
						
							|  |  |  | # 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.""" | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |     __slots__ = ('_path', '_drv', '_root', '_tail') | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, path): | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         self._path = path | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         self._drv = path.drive | 
					
						
							|  |  |  |         self._root = path.root | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         self._tail = path._tail | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __len__(self): | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         return len(self._tail) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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) | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         return self._path._from_parsed_parts(self._drv, self._root, | 
					
						
							|  |  |  |                                              self._tail[:-idx - 1]) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         return "<{}.parents>".format(type(self._path).__name__) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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. | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     __slots__ = ( | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         # The `_raw_path` slot stores an unnormalized string path. This is set | 
					
						
							|  |  |  |         # in the `__init__()` method. | 
					
						
							|  |  |  |         '_raw_path', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # The `_drv`, `_root` and `_tail_cached` slots store parsed and | 
					
						
							|  |  |  |         # normalized parts of the path. They are set when any of the `drive`, | 
					
						
							|  |  |  |         # `root` or `_tail` properties are accessed for the first time. The | 
					
						
							|  |  |  |         # three-part division corresponds to the result of | 
					
						
							|  |  |  |         # `os.path.splitroot()`, except that the tail is further split on path | 
					
						
							|  |  |  |         # separators (i.e. it is a list of strings), and that the root and | 
					
						
							|  |  |  |         # tail are normalized. | 
					
						
							|  |  |  |         '_drv', '_root', '_tail_cached', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # The `_str` slot stores the string representation of the path, | 
					
						
							|  |  |  |         # computed from the drive, root and tail when `__str__()` is called | 
					
						
							|  |  |  |         # for the first time. It's used to implement `_str_normcase` | 
					
						
							|  |  |  |         '_str', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # The `_str_normcase_cached` slot stores the string path with | 
					
						
							|  |  |  |         # normalized case. It is set when the `_str_normcase` property is | 
					
						
							|  |  |  |         # accessed for the first time. It's used to implement `__eq__()` | 
					
						
							|  |  |  |         # `__hash__()`, and `_parts_normcase` | 
					
						
							|  |  |  |         '_str_normcase_cached', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # The `_parts_normcase_cached` slot stores the case-normalized | 
					
						
							|  |  |  |         # string path after splitting on path separators. It's set when the | 
					
						
							|  |  |  |         # `_parts_normcase` property is accessed for the first time. It's used | 
					
						
							|  |  |  |         # to implement comparison methods like `__lt__()`. | 
					
						
							|  |  |  |         '_parts_normcase_cached', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # The `_hash` slot stores the hash of the case-normalized string | 
					
						
							|  |  |  |         # path. It's set when `__hash__()` is called for the first time. | 
					
						
							|  |  |  |         '_hash', | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |     def __new__(cls, *args, **kwargs): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """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 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         return object.__new__(cls) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __reduce__(self): | 
					
						
							|  |  |  |         # Using the parts tuple helps share interned path parts | 
					
						
							|  |  |  |         # when pickling related paths. | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         return (self.__class__, self.parts) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |     def __init__(self, *args): | 
					
						
							| 
									
										
										
										
											2023-05-02 19:08:19 +01:00
										 |  |  |         paths = [] | 
					
						
							|  |  |  |         for arg in args: | 
					
						
							|  |  |  |             if isinstance(arg, PurePath): | 
					
						
							|  |  |  |                 path = arg._raw_path | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     path = os.fspath(arg) | 
					
						
							|  |  |  |                 except TypeError: | 
					
						
							|  |  |  |                     path = arg | 
					
						
							|  |  |  |                 if not isinstance(path, str): | 
					
						
							|  |  |  |                     raise TypeError( | 
					
						
							|  |  |  |                         "argument should be a str or an os.PathLike " | 
					
						
							|  |  |  |                         "object where __fspath__ returns a str, " | 
					
						
							|  |  |  |                         f"not {type(path).__name__!r}") | 
					
						
							|  |  |  |             paths.append(path) | 
					
						
							|  |  |  |         if len(paths) == 0: | 
					
						
							|  |  |  |             self._raw_path = '' | 
					
						
							|  |  |  |         elif len(paths) == 1: | 
					
						
							|  |  |  |             self._raw_path = paths[0] | 
					
						
							| 
									
										
										
										
											2023-03-05 22:00:56 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2023-05-02 19:08:19 +01:00
										 |  |  |             self._raw_path = self._flavour.join(*paths) | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |     def with_segments(self, *pathsegments): | 
					
						
							|  |  |  |         """Construct a new path object from any number of path-like objects.
 | 
					
						
							|  |  |  |         Subclasses may override this method to customize how new path objects | 
					
						
							|  |  |  |         are created from methods like `iterdir()`. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return type(self)(*pathsegments) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def _parse_path(cls, path): | 
					
						
							|  |  |  |         if not path: | 
					
						
							|  |  |  |             return '', '', [] | 
					
						
							|  |  |  |         sep = cls._flavour.sep | 
					
						
							|  |  |  |         altsep = cls._flavour.altsep | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							| 
									
										
										
										
											2023-04-14 21:55:41 +01:00
										 |  |  |         if not root and drv.startswith(sep) and not drv.endswith(sep): | 
					
						
							|  |  |  |             drv_parts = drv.split(sep) | 
					
						
							|  |  |  |             if len(drv_parts) == 4 and drv_parts[2] not in '?.': | 
					
						
							|  |  |  |                 # e.g. //server/share | 
					
						
							|  |  |  |                 root = sep | 
					
						
							|  |  |  |             elif len(drv_parts) == 6: | 
					
						
							|  |  |  |                 # e.g. //?/unc/server/share | 
					
						
							|  |  |  |                 root = sep | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         parsed = [sys.intern(str(x)) for x in rel.split(sep) if x and x != '.'] | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         return drv, root, parsed | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |     def _load_parts(self): | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         drv, root, tail = self._parse_path(self._raw_path) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         self._drv = drv | 
					
						
							|  |  |  |         self._root = root | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         self._tail_cached = tail | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |     def _from_parsed_parts(self, drv, root, tail): | 
					
						
							|  |  |  |         path_str = self._format_parsed_parts(drv, root, tail) | 
					
						
							|  |  |  |         path = self.with_segments(path_str) | 
					
						
							|  |  |  |         path._str = path_str or '.' | 
					
						
							|  |  |  |         path._drv = drv | 
					
						
							|  |  |  |         path._root = root | 
					
						
							|  |  |  |         path._tail_cached = tail | 
					
						
							|  |  |  |         return path | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |     def _format_parsed_parts(cls, drv, root, tail): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         if drv or root: | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |             return drv + root + cls._flavour.sep.join(tail) | 
					
						
							|  |  |  |         elif tail and cls._flavour.splitdrive(tail[0])[0]: | 
					
						
							|  |  |  |             tail = ['.'] + tail | 
					
						
							|  |  |  |         return cls._flavour.sep.join(tail) | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |             self._str = self._format_parsed_parts(self.drive, self.root, | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |                                                   self._tail) or '.' | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         drive = self.drive | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def _str_normcase(self): | 
					
						
							|  |  |  |         # String with normalized case, for hashing and equality checks | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return self._str_normcase_cached | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             self._str_normcase_cached = self._flavour.normcase(str(self)) | 
					
						
							|  |  |  |             return self._str_normcase_cached | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     @property | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |     def _parts_normcase(self): | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         # Cached parts with normalized case, for comparisons. | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |             self._parts_normcase_cached = self._str_normcase.split(self._flavour.sep) | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             return self._parts_normcase_cached | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __eq__(self, other): | 
					
						
							|  |  |  |         if not isinstance(other, PurePath): | 
					
						
							|  |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         return self._str_normcase == other._str_normcase and self._flavour is other._flavour | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __hash__(self): | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return self._hash | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |             self._hash = hash(self._str_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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |     @property | 
					
						
							|  |  |  |     def drive(self): | 
					
						
							|  |  |  |         """The drive prefix (letter or UNC path), if any.""" | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return self._drv | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             self._load_parts() | 
					
						
							|  |  |  |             return self._drv | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def root(self): | 
					
						
							|  |  |  |         """The root of the path, if any.""" | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return self._root | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             self._load_parts() | 
					
						
							|  |  |  |             return self._root | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |     @property | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |     def _tail(self): | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |             return self._tail_cached | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         except AttributeError: | 
					
						
							|  |  |  |             self._load_parts() | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |             return self._tail_cached | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def anchor(self): | 
					
						
							|  |  |  |         """The concatenation of the drive and root, or ''.""" | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         anchor = self.drive + self.root | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         return anchor | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def name(self): | 
					
						
							|  |  |  |         """The final path component, if any.""" | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         tail = self._tail | 
					
						
							|  |  |  |         if not tail: | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return '' | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         return tail[-1] | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @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)) | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         return self._from_parsed_parts(self.drive, self.root, | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |                                        self._tail[:-1] + [name]) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         return self._from_parsed_parts(self.drive, self.root, | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |                                        self._tail[:-1] + [name]) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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)) | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         other = self.with_segments(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}") | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         parts = ['..'] * step + self._tail[len(path._tail):] | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         return self.with_segments(*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)) | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         other = self.with_segments(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."""
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         if self.drive or self.root: | 
					
						
							|  |  |  |             return (self.drive + self.root,) + tuple(self._tail) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return tuple(self._tail) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |     def joinpath(self, *pathsegments): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """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). | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         return self.with_segments(self, *pathsegments) | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |             return self.with_segments(key, self) | 
					
						
							| 
									
										
										
										
											2019-08-08 01:41:10 -04:00
										 |  |  |         except TypeError: | 
					
						
							|  |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def parent(self): | 
					
						
							|  |  |  |         """The logical parent of the path.""" | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         drv = self.drive | 
					
						
							|  |  |  |         root = self.root | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         tail = self._tail | 
					
						
							|  |  |  |         if not tail: | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             return self | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         return self._from_parsed_parts(drv, root, tail[:-1]) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def parents(self): | 
					
						
							|  |  |  |         """A sequence of this path's logical parents.""" | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         # The value of this property should not be cached on the path object, | 
					
						
							|  |  |  |         # as doing so would introduce a reference cycle. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         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: | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |             return bool(self.drive and self.root) | 
					
						
							| 
									
										
										
										
											2023-05-06 19:03:07 +01:00
										 |  |  |         return self._flavour.isabs(self._raw_path) | 
					
						
							| 
									
										
										
										
											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."""
 | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         if self._flavour is posixpath or not self._tail: | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             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. | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         if self.drive.startswith('\\\\'): | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |             # UNC paths are never reserved. | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         name = self._tail[-1].partition('.')[0].partition(':')[0].rstrip(' ') | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         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. | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         pat = self.with_segments(path_pattern) | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         if not pat.parts: | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             raise ValueError("empty pattern") | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         pat_parts = pat._parts_normcase | 
					
						
							| 
									
										
										
										
											2022-12-23 22:52:23 +00:00
										 |  |  |         parts = self._parts_normcase | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |         if pat.drive or pat.root: | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |             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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +01:00
										 |  |  |     def stat(self, *, follow_symlinks=True): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Return the result of the stat() system call on this path, like | 
					
						
							|  |  |  |         os.stat() does. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return os.stat(self, follow_symlinks=follow_symlinks) | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +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. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self.stat(follow_symlinks=False) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +01:00
										 |  |  |     # Convenience functions for querying the stat results | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +01:00
										 |  |  |     def exists(self, *, follow_symlinks=True): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path exists. | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +01:00
										 |  |  |         This method normally follows symlinks; to check whether a symlink exists, | 
					
						
							|  |  |  |         add the argument follow_symlinks=False. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             self.stat(follow_symlinks=follow_symlinks) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							|  |  |  |             if not _ignore_error(e): | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         return True | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +01:00
										 |  |  |     def is_dir(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a directory. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISDIR(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							|  |  |  |             if not _ignore_error(e): | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							|  |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +01:00
										 |  |  |     def is_file(self): | 
					
						
							| 
									
										
										
										
											2015-01-12 21:03:41 +01:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +01:00
										 |  |  |         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: | 
					
						
							|  |  |  |             if not _ignore_error(e): | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							|  |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_mount(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Check if this path is a mount point | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self._flavour.ismount(self) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_symlink(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a symbolic link. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISLNK(self.lstat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							|  |  |  |             if not _ignore_error(e): | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_junction(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a junction. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return self._flavour.isjunction(self) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_block_device(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a block device. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISBLK(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							|  |  |  |             if not _ignore_error(e): | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							|  |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_char_device(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a character device. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISCHR(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							|  |  |  |             if not _ignore_error(e): | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							|  |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_fifo(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a FIFO. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISFIFO(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							|  |  |  |             if not _ignore_error(e): | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							|  |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def is_socket(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Whether this path is a socket. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return S_ISSOCK(self.stat().st_mode) | 
					
						
							|  |  |  |         except OSError as e: | 
					
						
							|  |  |  |             if not _ignore_error(e): | 
					
						
							|  |  |  |                 raise | 
					
						
							|  |  |  |             # Path doesn't exist or is a broken symlink | 
					
						
							|  |  |  |             # (see http://web.archive.org/web/20200623061726/https://bitbucket.org/pitrou/pathlib/issues/12/ ) | 
					
						
							|  |  |  |             return False | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							|  |  |  |             # Non-encodable path | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |             other_st = self.with_segments(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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +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. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if "b" not in mode: | 
					
						
							|  |  |  |             encoding = io.text_encoding(encoding) | 
					
						
							|  |  |  |         return io.open(self, mode, buffering, encoding, errors, newline) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     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. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         encoding = io.text_encoding(encoding) | 
					
						
							|  |  |  |         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) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def write_text(self, data, encoding=None, errors=None, newline=None): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         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__) | 
					
						
							|  |  |  |         encoding = io.text_encoding(encoding) | 
					
						
							|  |  |  |         with self.open(mode='w', encoding=encoding, errors=errors, newline=newline) as f: | 
					
						
							|  |  |  |             return f.write(data) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +01:00
										 |  |  |     def _make_child_relpath(self, name): | 
					
						
							|  |  |  |         path_str = str(self) | 
					
						
							|  |  |  |         tail = self._tail | 
					
						
							|  |  |  |         if tail: | 
					
						
							|  |  |  |             path_str = f'{path_str}{self._flavour.sep}{name}' | 
					
						
							|  |  |  |         elif path_str != '.': | 
					
						
							|  |  |  |             path_str = f'{path_str}{name}' | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             path_str = name | 
					
						
							|  |  |  |         path = self.with_segments(path_str) | 
					
						
							|  |  |  |         path._str = path_str | 
					
						
							|  |  |  |         path._drv = self.drive | 
					
						
							|  |  |  |         path._root = self.root | 
					
						
							|  |  |  |         path._tail_cached = tail + [name] | 
					
						
							|  |  |  |         return path | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  |     def glob(self, pattern, *, case_sensitive=None): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """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)) | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         drv, root, pattern_parts = self._parse_path(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('') | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  |         selector = _make_selector(tuple(pattern_parts), self._flavour, case_sensitive) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         for p in selector.select_from(self): | 
					
						
							|  |  |  |             yield p | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  |     def rglob(self, pattern, *, case_sensitive=None): | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         """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) | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         drv, root, pattern_parts = self._parse_path(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('') | 
					
						
							| 
									
										
										
										
											2023-05-04 17:44:36 +01:00
										 |  |  |         selector = _make_selector(("**",) + tuple(pattern_parts), self._flavour, case_sensitive) | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |         for p in selector.select_from(self): | 
					
						
							|  |  |  |             yield p | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-07 20:07:07 +01: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) | 
					
						
							|  |  |  |         paths = [self] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while paths: | 
					
						
							|  |  |  |             path = paths.pop() | 
					
						
							|  |  |  |             if isinstance(path, tuple): | 
					
						
							|  |  |  |                 yield path | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # 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 in that instance, 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 = path._scandir() | 
					
						
							|  |  |  |             except OSError as error: | 
					
						
							|  |  |  |                 if on_error is not None: | 
					
						
							|  |  |  |                     on_error(error) | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             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 path, dirnames, filenames | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 paths.append((path, dirnames, filenames)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             paths += [path._make_child_relpath(d) for d in reversed(dirnames)] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, *args, **kwargs): | 
					
						
							|  |  |  |         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)) | 
					
						
							|  |  |  |         super().__init__(*args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __new__(cls, *args, **kwargs): | 
					
						
							|  |  |  |         if cls is Path: | 
					
						
							|  |  |  |             cls = WindowsPath if os.name == 'nt' else PosixPath | 
					
						
							|  |  |  |         return object.__new__(cls) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							|  |  |  |         # 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) | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __exit__(self, t, v, tb): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Public API | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def cwd(cls): | 
					
						
							|  |  |  |         """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() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def home(cls): | 
					
						
							|  |  |  |         """Return a new path pointing to the user's home directory (as
 | 
					
						
							|  |  |  |         returned by os.path.expanduser('~')). | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         return cls("~").expanduser() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 17:38:12 +01:00
										 |  |  |     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-04-03 19:57:11 +01:00
										 |  |  |         elif self.drive: | 
					
						
							| 
									
										
										
										
											2023-02-17 14:08:14 +00:00
										 |  |  |             # There is a CWD on each drive-letter drive. | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |             cwd = self._flavour.abspath(self.drive) | 
					
						
							| 
									
										
										
										
											2023-02-17 14:08:14 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             cwd = os.getcwd() | 
					
						
							| 
									
										
										
										
											2023-05-06 19:03:07 +01:00
										 |  |  |             # Fast path for "empty" paths, e.g. Path("."), Path("") or Path(). | 
					
						
							|  |  |  |             # We pass only one argument to with_segments() to avoid the cost | 
					
						
							|  |  |  |             # of joining, and we exploit the fact that getcwd() returns a | 
					
						
							|  |  |  |             # fully-normalized string by storing it in _str. This is used to | 
					
						
							|  |  |  |             # implement Path.cwd(). | 
					
						
							|  |  |  |             if not self.root and not self._tail: | 
					
						
							|  |  |  |                 result = self.with_segments(cwd) | 
					
						
							|  |  |  |                 result._str = cwd | 
					
						
							|  |  |  |                 return result | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         return self.with_segments(cwd, self) | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         p = self.with_segments(s) | 
					
						
							| 
									
										
										
										
											2021-04-28 16:50:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # 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
										 |  |  | 
 | 
					
						
							|  |  |  |     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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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") | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         return self.with_segments(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 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) | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         return self.with_segments(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) | 
					
						
							| 
									
										
										
										
											2023-05-05 20:04:53 +01:00
										 |  |  |         return self.with_segments(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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2023-04-03 19:57:11 +01:00
										 |  |  |         if (not (self.drive or self.root) and | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |             self._tail and self._tail[0][:1] == '~'): | 
					
						
							|  |  |  |             homedir = self._flavour.expanduser(self._tail[0]) | 
					
						
							| 
									
										
										
										
											2021-04-07 23:50:13 +01:00
										 |  |  |             if homedir[:1] == "~": | 
					
						
							|  |  |  |                 raise RuntimeError("Could not determine home directory.") | 
					
						
							| 
									
										
										
										
											2023-04-09 18:40:03 +01:00
										 |  |  |             drv, root, tail = self._parse_path(homedir) | 
					
						
							|  |  |  |             return self._from_parsed_parts(drv, root, tail + self._tail[1:]) | 
					
						
							| 
									
										
										
										
											2014-12-30 20:54:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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") |