Source code for baseband.base.frame

# Licensed under the GPLv3 - see LICENSE
"""
Base definitions for baseband frames.

Defines a frame class FrameBase that can be used to hold a header and a
payload, providing access to the values encoded in both.
"""
import numpy as np


__all__ = ['FrameBase']


[docs]class FrameBase: """Representation of a baseaband frame, consisting of a header and payload. Parameters ---------- header : ``cls._header_class`` Wrapper around the header, providing mechanisms to decode it. payload : ``cls._payload_class`` Wrapper around the payload, providing mechanisms to decode it. valid : bool, optional Whether the data are valid. Default: class default, `True`. verify : bool, optional Whether to do basic verification of integrity. Default: `True`. Notes ----- The Frame can also be instantiated using class methods: fromfile : read header and payload from a filehandle fromdata : encode data as payload Of course, one can also do the opposite: tofile : method to write header and payload to filehandle data : property that yields full decoded payload One can decode part of the payload by indexing or slicing the frame. If the frame does not contain valid data, all values returned are set to ``self.fill_value``. A number of properties are defined: `shape` and `dtype` are the shape and type of the data array, and `nbytes` the frame size in bytes. Furthermore, the frame acts as a dictionary, with keys those of the header. Any attribute that is not defined on the frame itself, such as ``.time`` will be looked up on the header as well. """ _header_class = None _payload_class = None _fill_value = 0. _valid = True def __init__(self, header, payload, valid=None, verify=True): self.header = header self.payload = payload if valid is not None: self.valid = valid if verify: self.verify()
[docs] def verify(self): """Simple verification. To be added to by subclasses.""" assert isinstance(self.header, self._header_class) assert isinstance(self.payload, self._payload_class) payload_nbytes = getattr(self.header, 'payload_nbytes', None) if payload_nbytes is not None: assert self.payload.nbytes == payload_nbytes
@property def valid(self): """Whether frame contains valid data.""" return self._valid @valid.setter def valid(self, valid): self._valid = bool(valid)
[docs] @classmethod def fromfile(cls, fh, memmap=None, valid=None, verify=True, **kwargs): """Read a frame from a filehandle. Parameters ---------- fh : filehandle Handle to read the frame from memmap : bool, optional If `False`, read payload from file. If `True`, map the payload in memory (see `~numpy.memmap`). Only useful for large payloads. Default: as set by payload class. valid : bool, optional Whether the data are valid. Default: inferred from header or payload read from file if possible, otherwise `True`. verify : bool, optional Whether to do basic verification of integrity. Default: `True`. **kwargs Extra arguments that help to initialize the payload. """ header = cls._header_class.fromfile(fh, verify=verify) payload = cls._payload_class.fromfile(fh, header=header, memmap=memmap, **kwargs) return cls(header, payload, valid=valid, verify=verify)
[docs] def tofile(self, fh): """Write encoded frame to filehandle.""" self.header.tofile(fh) self.payload.tofile(fh)
[docs] @classmethod def fromdata(cls, data, header=None, *, valid=None, verify=True, **kwargs): """Construct frame from data and header. Parameters ---------- data : `~numpy.ndarray` Array holding data to be encoded. header : header instance, optional Header for the frame. valid : bool, optional Whether the data are valid. Default: inferred from header if possible, otherwise `True`. verify : bool, optional Whether to verify the header and frame correctness. **kwargs Used to initialize the header, if not given. """ if header is None: header = cls._header_class.fromvalues(verify=verify, **kwargs) payload = cls._payload_class.fromdata(data, header=header) return cls(header, payload, valid=valid, verify=verify)
@property def sample_shape(self): """Shape of a sample in the frame (nchan,).""" return self.payload.sample_shape def __len__(self): """Number of samples in the frame.""" return len(self.payload) @property def shape(self): """Shape of the frame data.""" return (len(self),) + self.sample_shape @property def size(self): """Total number of component samples in the frame data.""" prod = 1 for dim in self.shape: prod *= dim return prod @property def ndim(self): """Number of dimensions of the frame data.""" return len(self.shape) @property def dtype(self): """Numeric type of the frame data.""" return self.payload.dtype @property def nbytes(self): """Size of the encoded frame in bytes.""" return self.header.nbytes + self.payload.nbytes @property def fill_value(self): """Value to replace invalid data in the frame.""" return self._fill_value @fill_value.setter def fill_value(self, fill_value): self._fill_value = fill_value def __array__(self, dtype=None): """Interface to arrays.""" if dtype is None or dtype == self.dtype: return self.data else: return self.data.astype(dtype) # Header behaves as a dictionary, while Payload can be indexed/sliced. # Let frame behave appropriately. def __getitem__(self, item=()): if isinstance(item, str): return self.header.__getitem__(item) elif self.valid: return self.payload.__getitem__(item) else: data_shape = np.empty(self.shape, dtype=bool)[item].shape return np.full(data_shape, self.fill_value, dtype=self.dtype) data = property(__getitem__, doc="Full decoded frame.") def __setitem__(self, item, value): if isinstance(item, str): self.header.__setitem__(item, value) else: self.payload.__setitem__(item, value)
[docs] def keys(self): return self.header.keys()
def _ipython_key_completions_(self): # Enables tab-completion of header keys in IPython. return self.header.keys() def __contains__(self, key): return key in self.keys() # Try to get any attribute not on the frame from the header properties. def __getattr__(self, attr): if (attr in self.header._properties or attr in ('get_time', 'set_time', 'update')): return getattr(self.header, attr) else: # Raise appropriate error. return self.__getattribute__(attr) def __setattr__(self, attr, value): if (attr not in {'header', 'payload', 'valid'} and attr in getattr(getattr(self, 'header', None), '_properties', ())): return setattr(self.header, attr, value) else: return super().__setattr__(attr, value) # For tests, it is useful to define equality. def __eq__(self, other): return (type(self) is type(other) and self.valid == other.valid and self.header == other.header and self.payload == other.payload)