Source code for ccsdspy.packet_fields

"""High level object-oriented API for elements that can exist in a packet.

See also:
- packet_types.FixedLength
- packet_types.VariableLength
"""

__author__ = "Daniel da Silva <mail@danieldasilva.org>"

import string

import numpy as np

from .converters import Converter


[docs] class PacketField: """A field contained in a packet.""" def __init__(self, name, data_type, bit_length, bit_offset=None, byte_order="big"): """ Parameters ---------- name : str String identifier for the field. The name specified how you may call upon this data later. data_type : {'uint', 'int', 'float', 'str', 'fill'} Data type of the field. bit_length : int Number of bits contained in the field. bit_offset : int, optional Bit offset into packet, including the primary header which is 48 bits long. If this is not specified, than the bit offset will the be calculated automatically from its position inside the packet definition. byte_order : {'big', 'little'}, or custom string of digits like "4321" Byte order of the field. Can be "big", "little", or an arbitrary ordering Specified as a string of digits like "2341". Defaults to big endian. Raises ------ TypeError If one of the arguments is not of the correct type. ValueError data_type or byte_order is invalid """ if not isinstance(name, str): raise TypeError("name parameter must be a str") if not isinstance(data_type, str): raise TypeError("data_type parameter must be a str") if not isinstance(bit_length, (int, np.integer)): raise TypeError("bit_length parameter must be an int") if not (bit_offset is None or isinstance(bit_offset, (int, np.integer))): raise TypeError("bit_offset parameter must be an int") valid_data_types = ("uint", "int", "float", "str", "fill") if data_type not in valid_data_types: raise ValueError(f"data_type must be one of {valid_data_types}") valid_default_byte_orders = ("big", "little") valid_custom_byte_order = all(char in string.digits for char in byte_order) if byte_order in valid_default_byte_orders: byte_order_parse = byte_order byte_order_post = None elif valid_custom_byte_order: byte_order_parse = "big" byte_order_post = byte_order else: raise ValueError( f"byte_order must be one of {valid_default_byte_orders} or " "a string like '1234' or '3412'." ) self._name = name self._data_type = data_type self._bit_length = bit_length self._bit_offset = bit_offset self._byte_order = byte_order self._byte_order_parse = byte_order_parse self._byte_order_post = byte_order_post self._field_type = "element" self._array_shape = None self._array_order = None def __repr__(self): values = {k: repr(v) for (k, v) in self.__dict__.items()} values["cls_name"] = self.__class__.__name__ if values["cls_name"] == "PacketArray": return ( "{cls_name}(name={_name}, data_type={_data_type}, " "bit_length={_bit_length}, bit_offset={_bit_offset}, " "byte_order={_byte_order}, array_shape={_array_shape}, " "array_order={_array_order})".format(**values) ) else: return ( "{cls_name}(name={_name}, data_type={_data_type}, " "bit_length={_bit_length}, bit_offset={_bit_offset}, " "byte_order={_byte_order})".format(**values) ) def __iter__(self): return iter( [ ("name", self._name), ("dataType", self._data_type), ("bitLength", self._bit_length), ("bitOffset", self._bit_offset), ("byteOrder", self._byte_order), ] )
[docs] class PacketArray(PacketField): """An array contained in a packet, similar to :py:class:`~ccsdspy.PacketField` but with multiple elements of the same size (e.g. image). """ def __init__(self, *args, array_shape=None, array_order="C", **kwargs): """ Parameters ---------- name : str String identifier for the field. The name specified how you may call upon this data later. data_type : {'uint', 'int', 'float', 'str', 'fill'} Data type of the field. bit_length : int Number of bits contained in the field. array_shape : int, tuple of ints, str, 'expand' Shape of the array as a tuple. For a 1-dimensional array, a single integer can be supplied. To use another field's value, pass the name of that field. To grow to fill the packet, use "expand". For details on variable length fields, see the :py:class:`~ccsdspy.VariableLength` class. array_order {'C', 'F'} Row-major (C-style) or column-major (Fortran-style) order. bit_offset : int, optional Bit offset into packet, including the primary header which is 48 bits long. If this is not specified, than the bit offset will the be calculated automatically from its position inside the packet definition. byte_order : {'big', 'little'}, optional Byte order of the field. Defaults to big endian. Raises ------ TypeError If one of the arguments is not of the correct type. ValueError array_shape, array_order, data_type, or byte_order is invalid """ if isinstance(array_shape, str): if "data_type" not in kwargs: kwargs["data_type"] = "uint" elif kwargs["data_type"] != "uint": raise ValueError("Expanding arrays must be data_type='uint'") else: if isinstance(array_shape, int): array_shape = (array_shape,) if not isinstance(array_shape, tuple): raise TypeError("array_shape parameter must be a tuple of ints") if not all(isinstance(dim, int) for dim in array_shape): raise TypeError("array_shape parameter must be a tuple of ints") if not all(dim >= 0 for dim in array_shape): raise TypeError("array_shape parameter dimensions must be >= 0") if sum(array_shape) == 0: raise TypeError("array must have at least one element") if not isinstance(array_order, str): raise TypeError("array_order parameter must be string") if array_order not in {"C", "F"}: raise TypeError("array_order parameter must be either 'C' or 'F'") super().__init__(*args, **kwargs) self._field_type = "array" self._array_shape = array_shape self._array_order = array_order