mirror of
https://github.com/python/cpython.git
synced 2025-10-24 18:33:49 +00:00
gh-91298: Refine traversable (apply changes from importlib_resources 5.7.1) (#91623)
* bpo-47142: Refine traversable (apply changes from importlib_resources 5.7.1) * Replace changelog referencing github issue.
This commit is contained in:
parent
67712e71b3
commit
7659681556
4 changed files with 42 additions and 16 deletions
|
@ -1,6 +1,14 @@
|
||||||
import abc
|
import abc
|
||||||
from typing import BinaryIO, Iterable, Text
|
import io
|
||||||
|
import os
|
||||||
|
from typing import Any, BinaryIO, Iterable, Iterator, NoReturn, Text, Optional
|
||||||
from typing import runtime_checkable, Protocol
|
from typing import runtime_checkable, Protocol
|
||||||
|
from typing import Union
|
||||||
|
|
||||||
|
|
||||||
|
StrPath = Union[str, os.PathLike[str]]
|
||||||
|
|
||||||
|
__all__ = ["ResourceReader", "Traversable", "TraversableResources"]
|
||||||
|
|
||||||
|
|
||||||
class ResourceReader(metaclass=abc.ABCMeta):
|
class ResourceReader(metaclass=abc.ABCMeta):
|
||||||
|
@ -50,22 +58,25 @@ class Traversable(Protocol):
|
||||||
"""
|
"""
|
||||||
An object with a subset of pathlib.Path methods suitable for
|
An object with a subset of pathlib.Path methods suitable for
|
||||||
traversing directories and opening files.
|
traversing directories and opening files.
|
||||||
|
|
||||||
|
Any exceptions that occur when accessing the backing resource
|
||||||
|
may propagate unaltered.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def iterdir(self):
|
def iterdir(self) -> Iterator["Traversable"]:
|
||||||
"""
|
"""
|
||||||
Yield Traversable objects in self
|
Yield Traversable objects in self
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def read_bytes(self):
|
def read_bytes(self) -> bytes:
|
||||||
"""
|
"""
|
||||||
Read contents of self as bytes
|
Read contents of self as bytes
|
||||||
"""
|
"""
|
||||||
with self.open('rb') as strm:
|
with self.open('rb') as strm:
|
||||||
return strm.read()
|
return strm.read()
|
||||||
|
|
||||||
def read_text(self, encoding=None):
|
def read_text(self, encoding: Optional[str] = None) -> str:
|
||||||
"""
|
"""
|
||||||
Read contents of self as text
|
Read contents of self as text
|
||||||
"""
|
"""
|
||||||
|
@ -85,12 +96,16 @@ def is_file(self) -> bool:
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def joinpath(self, child):
|
def joinpath(self, *descendants: StrPath) -> "Traversable":
|
||||||
"""
|
"""
|
||||||
Return Traversable child in self
|
Return Traversable resolved with any descendants applied.
|
||||||
|
|
||||||
|
Each descendant should be a path segment relative to self
|
||||||
|
and each may contain multiple levels separated by
|
||||||
|
``posixpath.sep`` (``/``).
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __truediv__(self, child):
|
def __truediv__(self, child: StrPath) -> "Traversable":
|
||||||
"""
|
"""
|
||||||
Return Traversable child in self
|
Return Traversable child in self
|
||||||
"""
|
"""
|
||||||
|
@ -120,17 +135,17 @@ class TraversableResources(ResourceReader):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def files(self):
|
def files(self) -> "Traversable":
|
||||||
"""Return a Traversable object for the loaded package."""
|
"""Return a Traversable object for the loaded package."""
|
||||||
|
|
||||||
def open_resource(self, resource):
|
def open_resource(self, resource: StrPath) -> io.BufferedReader:
|
||||||
return self.files().joinpath(resource).open('rb')
|
return self.files().joinpath(resource).open('rb')
|
||||||
|
|
||||||
def resource_path(self, resource):
|
def resource_path(self, resource: Any) -> NoReturn:
|
||||||
raise FileNotFoundError(resource)
|
raise FileNotFoundError(resource)
|
||||||
|
|
||||||
def is_resource(self, path):
|
def is_resource(self, path: StrPath) -> bool:
|
||||||
return self.files().joinpath(path).is_file()
|
return self.files().joinpath(path).is_file()
|
||||||
|
|
||||||
def contents(self):
|
def contents(self) -> Iterator[str]:
|
||||||
return (item.name for item in self.files().iterdir())
|
return (item.name for item in self.files().iterdir())
|
||||||
|
|
|
@ -99,10 +99,19 @@ def iterdir(self):
|
||||||
def open(self, *args, **kwargs):
|
def open(self, *args, **kwargs):
|
||||||
raise IsADirectoryError()
|
raise IsADirectoryError()
|
||||||
|
|
||||||
def joinpath(self, name):
|
@staticmethod
|
||||||
|
def _flatten(compound_names):
|
||||||
|
for name in compound_names:
|
||||||
|
yield from name.split('/')
|
||||||
|
|
||||||
|
def joinpath(self, *descendants):
|
||||||
|
if not descendants:
|
||||||
|
return self
|
||||||
|
names = self._flatten(descendants)
|
||||||
|
target = next(names)
|
||||||
return next(
|
return next(
|
||||||
traversable for traversable in self.iterdir() if traversable.name == name
|
traversable for traversable in self.iterdir() if traversable.name == target
|
||||||
)
|
).joinpath(*names)
|
||||||
|
|
||||||
|
|
||||||
class TraversableReader(TraversableResources, SimpleReader):
|
class TraversableReader(TraversableResources, SimpleReader):
|
||||||
|
|
|
@ -42,7 +42,7 @@ def generate(suffix):
|
||||||
|
|
||||||
def walk(datapath):
|
def walk(datapath):
|
||||||
for dirpath, dirnames, filenames in os.walk(datapath):
|
for dirpath, dirnames, filenames in os.walk(datapath):
|
||||||
with contextlib.suppress(KeyError):
|
with contextlib.suppress(ValueError):
|
||||||
dirnames.remove('__pycache__')
|
dirnames.remove('__pycache__')
|
||||||
for filename in filenames:
|
for filename in filenames:
|
||||||
res = pathlib.Path(dirpath) / filename
|
res = pathlib.Path(dirpath) / filename
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
In ``importlib.resources.abc``, refined the documentation of the Traversable
|
||||||
|
Protocol, applying changes from importlib_resources 5.7.1.
|
Loading…
Add table
Add a link
Reference in a new issue