| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| 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() |
|
|