summaryrefslogtreecommitdiffstats
path: root/contrib/python/fonttools/fontTools/misc/filesystem/_subfs.py
diff options
context:
space:
mode:
authorrobot-piglet <[email protected]>2025-07-31 11:14:11 +0300
committerrobot-piglet <[email protected]>2025-07-31 12:10:37 +0300
commite177928be72df9669dbb830824b4233a33c8723f (patch)
treea91d4ec6bbe7dc221c049475a91255c2996fd84e /contrib/python/fonttools/fontTools/misc/filesystem/_subfs.py
parenta1700abf3c749b43117e757deb259d2a7bcdf46a (diff)
Intermediate changes
commit_hash:60aaacde4a6a0fb68b6435d7f100365d0c77d64d
Diffstat (limited to 'contrib/python/fonttools/fontTools/misc/filesystem/_subfs.py')
-rw-r--r--contrib/python/fonttools/fontTools/misc/filesystem/_subfs.py92
1 files changed, 92 insertions, 0 deletions
diff --git a/contrib/python/fonttools/fontTools/misc/filesystem/_subfs.py b/contrib/python/fonttools/fontTools/misc/filesystem/_subfs.py
new file mode 100644
index 00000000000..bc7d8825af1
--- /dev/null
+++ b/contrib/python/fonttools/fontTools/misc/filesystem/_subfs.py
@@ -0,0 +1,92 @@
+from __future__ import annotations
+
+import typing
+from pathlib import PurePosixPath
+
+from ._base import FS
+from ._errors import DirectoryExpected, ResourceNotFound
+
+if typing.TYPE_CHECKING:
+ from collections.abc import Collection
+ from typing import IO, Any
+
+ from ._info import Info
+
+
+class SubFS(FS):
+ """Maps a sub-directory of another filesystem."""
+
+ def __init__(self, parent: FS, sub_path: str):
+ super().__init__()
+ self._parent = parent
+ self._prefix = PurePosixPath(sub_path).as_posix().rstrip("/")
+ if not parent.exists(self._prefix):
+ raise ResourceNotFound(f"No such file or directory: {sub_path!r}")
+ elif not parent.isdir(self._prefix):
+ raise DirectoryExpected(f"{sub_path!r} is not a directory")
+
+ def delegate_fs(self):
+ return self._parent
+
+ def _full(self, rel: str) -> str:
+ self.check()
+ return f"{self._prefix}/{PurePosixPath(rel).as_posix()}".lstrip("/")
+
+ def open(self, path: str, mode: str = "rb", **kwargs) -> IO[Any]:
+ return self._parent.open(self._full(path), mode, **kwargs)
+
+ def exists(self, path: str) -> bool:
+ return self._parent.exists(self._full(path))
+
+ def isdir(self, path: str) -> bool:
+ return self._parent.isdir(self._full(path))
+
+ def isfile(self, path: str) -> bool:
+ return self._parent.isfile(self._full(path))
+
+ def listdir(self, path: str) -> list[str]:
+ return self._parent.listdir(self._full(path))
+
+ def makedir(self, path: str, recreate: bool = False):
+ return self._parent.makedir(self._full(path), recreate=recreate)
+
+ def makedirs(self, path: str, recreate: bool = False):
+ return self._parent.makedirs(self._full(path), recreate=recreate)
+
+ def getinfo(self, path: str, namespaces: Collection[str] | None = None) -> Info:
+ return self._parent.getinfo(self._full(path), namespaces=namespaces)
+
+ def remove(self, path: str):
+ return self._parent.remove(self._full(path))
+
+ def removedir(self, path: str):
+ return self._parent.removedir(self._full(path))
+
+ def removetree(self, path: str):
+ return self._parent.removetree(self._full(path))
+
+ def movedir(self, src: str, dst: str, create: bool = False):
+ self._parent.movedir(self._full(src), self._full(dst), create=create)
+
+ def getsyspath(self, path: str) -> str:
+ return self._parent.getsyspath(self._full(path))
+
+ def readbytes(self, path: str) -> bytes:
+ return self._parent.readbytes(self._full(path))
+
+ def writebytes(self, path: str, data: bytes):
+ self._parent.writebytes(self._full(path), data)
+
+ def __repr__(self) -> str:
+ return f"{self.__class__.__name__}({self._parent!r}, {self._prefix!r})"
+
+ def __str__(self) -> str:
+ return f"{self._parent}/{self._prefix}"
+
+
+class ClosingSubFS(SubFS):
+ """Like SubFS, but auto-closes the parent filesystem when closed."""
+
+ def close(self):
+ super().close()
+ self._parent.close()