2016-02-26 10:05:51 -08:00
|
|
|
"""Base or helper classes used a lot for dealing with file formats.
|
|
|
|
"""
|
|
|
|
import io
|
|
|
|
import struct
|
|
|
|
|
|
|
|
|
|
|
|
class Substream:
|
|
|
|
"""Wraps a stream and pretends it starts at an offset other than 0.
|
|
|
|
|
|
|
|
Partly implements the file interface.
|
|
|
|
|
|
|
|
This type always seeks before reading, but doesn't do so afterwards, so
|
|
|
|
interleaving reads with the underlying stream may not do what you want.
|
|
|
|
"""
|
|
|
|
def __init__(self, stream, offset=0, length=-1):
|
|
|
|
if isinstance(stream, Substream):
|
|
|
|
self.stream = stream.stream
|
|
|
|
self.offset = offset + stream.offset
|
|
|
|
else:
|
|
|
|
self.stream = stream
|
|
|
|
self.offset = offset
|
|
|
|
|
|
|
|
self.length = length
|
|
|
|
self.pos = 0
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
return "<{} of {} at {}>".format(
|
|
|
|
type(self).__name__, self.stream, self.offset)
|
|
|
|
|
|
|
|
def read(self, n=-1):
|
|
|
|
self.stream.seek(self.offset + self.pos)
|
2016-12-20 19:20:33 -08:00
|
|
|
maxread = self.length - self.pos
|
|
|
|
if n < 0 or 0 <= maxread < n:
|
|
|
|
n = maxread
|
2016-02-26 10:05:51 -08:00
|
|
|
data = self.stream.read(n)
|
|
|
|
self.pos += len(data)
|
|
|
|
return data
|
|
|
|
|
2016-12-20 19:20:33 -08:00
|
|
|
def seek(self, offset, whence=0):
|
|
|
|
if whence == 1:
|
|
|
|
offset += self.pos
|
|
|
|
elif whence == 2:
|
|
|
|
offset += self.length
|
2016-02-26 10:05:51 -08:00
|
|
|
offset = max(offset, 0)
|
|
|
|
if self.length >= 0:
|
|
|
|
offset = min(offset, self.length)
|
|
|
|
self.stream.seek(self.offset + offset)
|
|
|
|
self.pos = self.tell()
|
|
|
|
|
|
|
|
def tell(self):
|
|
|
|
return self.stream.tell() - self.offset
|
|
|
|
|
|
|
|
def __len__(self):
|
|
|
|
if self.length < 0:
|
|
|
|
pos = self.stream.tell()
|
|
|
|
self.stream.seek(0, io.SEEK_END)
|
|
|
|
parent_length = self.stream.tell()
|
|
|
|
self.stream.seek(pos)
|
|
|
|
return parent_length - self.offset
|
|
|
|
else:
|
|
|
|
return self.length
|
|
|
|
|
|
|
|
def peek(self, n):
|
|
|
|
pos = self.stream.tell()
|
|
|
|
self.stream.seek(self.offset + self.pos)
|
2016-12-20 19:20:33 -08:00
|
|
|
maxread = self.length - self.pos
|
|
|
|
data = self.stream.read(min(maxread, n))
|
2016-02-26 10:05:51 -08:00
|
|
|
self.stream.seek(pos)
|
|
|
|
return data
|
|
|
|
|
|
|
|
def unpack(self, fmt):
|
|
|
|
"""Unpacks a struct format from the current position in the stream."""
|
|
|
|
data = self.read(struct.calcsize(fmt))
|
|
|
|
return struct.unpack(fmt, data)
|
|
|
|
|
|
|
|
def slice(self, offset, length=-1):
|
|
|
|
# TODO limit or warn if length is too long for this slice?
|
2016-12-20 19:20:33 -08:00
|
|
|
return Substream(self, offset, length)
|
2016-02-26 10:05:51 -08:00
|
|
|
|
|
|
|
|
|
|
|
class _ContainerFile:
|
|
|
|
slices = ()
|
|
|
|
|
|
|
|
def __len__(self):
|
|
|
|
return len(self.slices)
|
|
|
|
|
|
|
|
def __iter__(self):
|
|
|
|
return iter(self.slices)
|
|
|
|
|
|
|
|
def __getitem__(self, key):
|
|
|
|
return self.slices[key]
|