Source code for electrical_protocol.packet

from __future__ import annotations

import struct
from dataclasses import dataclass, fields
from enum import Enum
from functools import lru_cache
from typing import ClassVar, TypeVar, get_type_hints

SYNC_CHAR_1 = 0x37
SYNC_CHAR_2 = 0x01

_packet_registry: dict[int, dict[int, type[Packet]]] = {}


PacketSelf = TypeVar("PacketSelf", bound="Packet")


def hexify(data: bytes) -> str:
    return ":".join(f"{c:02x}" for c in data)


@lru_cache(maxsize=None)
def get_cache_hints(cls):
    return get_type_hints(cls)


[docs]class ChecksumException(OSError): """ An invalid checksum appeared. """ def __init__( self, packet: type[Packet], received: tuple[int, int], expected: tuple[int, int], ): """ Attributes: packet (Type[:class:`~.Packet`]): The packet with the invalid checksum. received (Tuple[int, int]): The received Fletcher's checksum. expected (Tuple[int, int]): The expected Fletcher's checksum. """ super().__init__( f"Invalid checksum in packet of type {packet.__qualname__}: received {received}, expected {expected}", )
[docs]@dataclass class Packet: """ Represents one packet that can be sent or received by a serial device. This class is able to handle packaging unique data values into a :class:`bytes` object for sending over a data stream. This class should be overridden to implement unique packet payloads. Note that this class supports three subclass arguments to assign unique message IDs, subclass IDs, and payload formats. Note that all subclasses must be decorated with :meth:`dataclasses.dataclass`. If any class members are annotated with a subclass of :class:`enum.Enum`, the class will always make an attempt to convert the raw data value to an instance of the enum before constructing the rest of the values in the class. .. code-block:: python from dataclasses import dataclass @dataclass class ExamplePacket(Packet, class_id = 0x02, subclass_id = 0x01, payload_format = "BHHf"): example_char: int example_short: int example_short_two: int example_float: float .. container:: operations .. describe:: bytes(x) Returns a :class:`bytes` object representing the data of the packet in the specified packet format. Arguments: class_id (int): The message ID. Can be between 0 and 255. subclass_id (int): The message subclass ID. Can be between 0 and 255. payload_format (str): The format for the payload. This determines how the individual payload is assembled. Each character in the format string represents the position of one class variable. The class variables are assembled in the order they are defined in. """ class_id: ClassVar[int] subclass_id: ClassVar[int] payload_format: ClassVar[str] = "" def __init_subclass__( cls, class_id: int, subclass_id: int, payload_format: str = "", ): cls.class_id = class_id cls.subclass_id = subclass_id cls.payload_format = payload_format packets = [p for mid in _packet_registry.values() for p in mid.values()] for packet in packets: if packet.class_id == class_id and packet.subclass_id == subclass_id: raise ValueError( f"Cannot reuse class_id 0x{class_id:0x} and subclass_id 0x{subclass_id}, already used by {packet.__qualname__}", ) _packet_registry.setdefault(class_id, {})[subclass_id] = cls def __post_init__(self): for name, field_type in get_cache_hints(self.__class__).items(): if name not in [ "class_id", "subclass_id", "payload_format", ] and not isinstance(self.__dict__[name], field_type): if issubclass(field_type, Enum): setattr(self, name, field_type(self.__dict__[name])) elif issubclass(field_type, str): setattr(self, name, self.__dict__[name].rstrip(b"\x00").decode()) if self.payload_format and not self.payload_format.startswith( ("<", ">", "=", "!"), ): raise ValueError( "The payload format does not start with a standard size character: ('<', '>', '!', '=').", ) available_chars: dict[type, list[str]] = { bool: ["c", "b", "B", "?"], int: ["b", "B", "h", "H", "i", "I", "l", "L", "q", "Q"], float: ["f", "d"], } stripped_format = self.payload_format.lstrip("<>=!@") for i, field in enumerate(fields(self)): if field.type not in available_chars: continue chars = available_chars[field.type] if i >= len(stripped_format): raise ValueError( f"The payload format for the packet is too short to support all dataclass fields; expected: {len(fields(self))}, found: {len(self.payload_format)}.", ) represented_char = stripped_format[i] if represented_char not in chars: raise ValueError( f"The type of {field.name} in the payload format is '{represented_char}', which does not correspond to its dataclass type of {field.type}.", ) @classmethod def _calculate_checksum(cls, data: bytes) -> tuple[int, int]: """ Used to calculate the Fletcher's checksum for a series of bytes. When calculating the checksum for a new packet, the start bytes/sync characters should not be included. """ sum1, sum2 = 0, 0 for byte in data: sum1 = (sum1 + byte) % 255 sum2 = (sum2 + sum1) % 255 return sum1, sum2 def __bytes__(self): ready_values = [] for value in self.__dict__.values(): if isinstance(value, Enum): ready_values.append( ( value.value if not isinstance(value.value, str) else value.value.encode() ), ) elif isinstance(value, str): ready_values.append(value.encode()) else: ready_values.append(value) payload = struct.pack(self.payload_format, *ready_values) data = struct.pack( f"<BBBBH{len(payload)}s", SYNC_CHAR_1, SYNC_CHAR_2, self.class_id, self.subclass_id, len(payload), payload, ) checksum = self._calculate_checksum(data[2:]) return data + struct.pack("<BB", *checksum) def __len__(self) -> int: return self.__class__._expected_len() @classmethod def _expected_len(cls) -> int: # We cannot use one calcsize since payload_format should start with a standard size character return struct.calcsize("<BBBBHBB") + struct.calcsize(cls.payload_format)
[docs] @classmethod def from_bytes( cls: type[PacketSelf], packed: bytes, trim: bool = True, ) -> PacketSelf: """ Constructs a packet from a packed packet in a :class:`bytes` object. If a packet is found with the corresponding message and subclass ID, then an instance (or subclass) of that packet class will be returned. Arguments: packed (bytes): The packed packet to unpack. trim (bool): If True, only the required number of bytes will be used to construct the packet. Otherwise, the entire packet will be used. Default: `True`. Raises: ChecksumException: The checksum is invalid. LookupError: No packet with the specified class and subclass IDs exist. Returns: An instance of the appropriate packet subclass. """ class_id = packed[2] subclass_id = packed[3] if class_id in _packet_registry and subclass_id in _packet_registry[class_id]: subclass = _packet_registry[class_id][subclass_id] if trim: packed = packed[: subclass._expected_len()] if len(packed) < subclass._expected_len(): raise ValueError( f"Packet is too short to be a valid packet. (provided len: {len(packed)}, expected len: {cls._expected_len()})", ) payload = packed[6:-2] if struct.unpack("<BB", packed[-2:]) != cls._calculate_checksum( packed[2:-2], ): raise ChecksumException( subclass, struct.unpack("<BB", packed[-2:]), cls._calculate_checksum(packed[2:-2]), ) unpacked = struct.unpack(subclass.payload_format, payload) packet = subclass(*unpacked) if not isinstance(packet, cls): raise RuntimeError( f"Attempted to resolve packet of type {cls.__qualname__}, but found {packet.__class__.__qualname__} for bytes: {hexify(packed)}", ) return packet raise LookupError( f"Attempted to reconstruct packet with class_id 0x{class_id:02x} and subclass_id 0x{subclass_id:02x}, but no packet with IDs was found.", )
[docs]@dataclass class AckPacket(Packet, class_id=0x00, subclass_id=0x01, payload_format=""): """ Common acknowledgment packet. """
[docs]@dataclass class NackPacket(Packet, class_id=0x00, subclass_id=0x00, payload_format=""): """ Common not-acknowledged packet. """