|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| from __future__ import annotations
|
|
|
| import io
|
| from collections.abc import Iterable
|
| from typing import IO, AnyStr, NoReturn
|
|
|
|
|
| class ContainerIO(IO[AnyStr]):
|
| """
|
| A file object that provides read access to a part of an existing
|
| file (for example a TAR file).
|
| """
|
|
|
| def __init__(self, file: IO[AnyStr], offset: int, length: int) -> None:
|
| """
|
| Create file object.
|
|
|
| :param file: Existing file.
|
| :param offset: Start of region, in bytes.
|
| :param length: Size of region, in bytes.
|
| """
|
| self.fh: IO[AnyStr] = file
|
| self.pos = 0
|
| self.offset = offset
|
| self.length = length
|
| self.fh.seek(offset)
|
|
|
|
|
|
|
|
|
| def isatty(self) -> bool:
|
| return False
|
|
|
| def seekable(self) -> bool:
|
| return True
|
|
|
| def seek(self, offset: int, mode: int = io.SEEK_SET) -> int:
|
| """
|
| Move file pointer.
|
|
|
| :param offset: Offset in bytes.
|
| :param mode: Starting position. Use 0 for beginning of region, 1
|
| for current offset, and 2 for end of region. You cannot move
|
| the pointer outside the defined region.
|
| :returns: Offset from start of region, in bytes.
|
| """
|
| if mode == 1:
|
| self.pos = self.pos + offset
|
| elif mode == 2:
|
| self.pos = self.length + offset
|
| else:
|
| self.pos = offset
|
|
|
| self.pos = max(0, min(self.pos, self.length))
|
| self.fh.seek(self.offset + self.pos)
|
| return self.pos
|
|
|
| def tell(self) -> int:
|
| """
|
| Get current file pointer.
|
|
|
| :returns: Offset from start of region, in bytes.
|
| """
|
| return self.pos
|
|
|
| def readable(self) -> bool:
|
| return True
|
|
|
| def read(self, n: int = -1) -> AnyStr:
|
| """
|
| Read data.
|
|
|
| :param n: Number of bytes to read. If omitted, zero or negative,
|
| read until end of region.
|
| :returns: An 8-bit string.
|
| """
|
| if n > 0:
|
| n = min(n, self.length - self.pos)
|
| else:
|
| n = self.length - self.pos
|
| if n <= 0:
|
| return b"" if "b" in self.fh.mode else ""
|
| self.pos = self.pos + n
|
| return self.fh.read(n)
|
|
|
| def readline(self, n: int = -1) -> AnyStr:
|
| """
|
| Read a line of text.
|
|
|
| :param n: Number of bytes to read. If omitted, zero or negative,
|
| read until end of line.
|
| :returns: An 8-bit string.
|
| """
|
| s: AnyStr = b"" if "b" in self.fh.mode else ""
|
| newline_character = b"\n" if "b" in self.fh.mode else "\n"
|
| while True:
|
| c = self.read(1)
|
| if not c:
|
| break
|
| s = s + c
|
| if c == newline_character or len(s) == n:
|
| break
|
| return s
|
|
|
| def readlines(self, n: int | None = -1) -> list[AnyStr]:
|
| """
|
| Read multiple lines of text.
|
|
|
| :param n: Number of lines to read. If omitted, zero, negative or None,
|
| read until end of region.
|
| :returns: A list of 8-bit strings.
|
| """
|
| lines = []
|
| while True:
|
| s = self.readline()
|
| if not s:
|
| break
|
| lines.append(s)
|
| if len(lines) == n:
|
| break
|
| return lines
|
|
|
| def writable(self) -> bool:
|
| return False
|
|
|
| def write(self, b: AnyStr) -> NoReturn:
|
| raise NotImplementedError()
|
|
|
| def writelines(self, lines: Iterable[AnyStr]) -> NoReturn:
|
| raise NotImplementedError()
|
|
|
| def truncate(self, size: int | None = None) -> int:
|
| raise NotImplementedError()
|
|
|
| def __enter__(self) -> ContainerIO[AnyStr]:
|
| return self
|
|
|
| def __exit__(self, *args: object) -> None:
|
| self.close()
|
|
|
| def __iter__(self) -> ContainerIO[AnyStr]:
|
| return self
|
|
|
| def __next__(self) -> AnyStr:
|
| line = self.readline()
|
| if not line:
|
| msg = "end of region"
|
| raise StopIteration(msg)
|
| return line
|
|
|
| def fileno(self) -> int:
|
| return self.fh.fileno()
|
|
|
| def flush(self) -> None:
|
| self.fh.flush()
|
|
|
| def close(self) -> None:
|
| self.fh.close()
|
|
|