|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
from __future__ import annotations |
|
|
|
import io |
|
from typing import IO, AnyStr, Generic, Literal |
|
|
|
|
|
class ContainerIO(Generic[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 seek(self, offset: int, mode: Literal[0, 1, 2] = io.SEEK_SET) -> None: |
|
""" |
|
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. |
|
""" |
|
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) |
|
|
|
def tell(self) -> int: |
|
""" |
|
Get current file pointer. |
|
|
|
:returns: Offset from start of region, in bytes. |
|
""" |
|
return self.pos |
|
|
|
def read(self, n: int = 0) -> AnyStr: |
|
""" |
|
Read data. |
|
|
|
:param n: Number of bytes to read. If omitted or zero, |
|
read until end of region. |
|
:returns: An 8-bit string. |
|
""" |
|
if n: |
|
n = min(n, self.length - self.pos) |
|
else: |
|
n = self.length - self.pos |
|
if not n: |
|
return b"" if "b" in self.fh.mode else "" |
|
self.pos = self.pos + n |
|
return self.fh.read(n) |
|
|
|
def readline(self) -> AnyStr: |
|
""" |
|
Read a line of text. |
|
|
|
: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: |
|
break |
|
return s |
|
|
|
def readlines(self) -> list[AnyStr]: |
|
""" |
|
Read multiple lines of text. |
|
|
|
:returns: A list of 8-bit strings. |
|
""" |
|
lines = [] |
|
while True: |
|
s = self.readline() |
|
if not s: |
|
break |
|
lines.append(s) |
|
return lines |
|
|