Source code for mwings.parsers.app_uart_ascii

# -*- coding:utf-8 -*-
# Written for Python 3.12
# Formatted with Black

# Packet parser for App_Uart (Mode A)

from datetime import datetime
from base64 import b64encode
from typing import Any, final

from pydantic import Field, computed_field, field_validator
from overrides import override

from .. import common


[docs] @final class ParsedPacket(common.ParsedPacketBase): """Dataclass for parsed packets from App_Uart (Mode A) Attributes ---------- command_id: common.UInt8 Command ID data: bytes Data body (Hidden in JSON or something) data_base64: str Data body in Base64 for JSON or something data_hexstr: str Data body in ASCII string """ command_id: common.UInt8 = Field(default=0x00, ge=0x00, lt=0x80) data: bytes = Field(default=bytes(), exclude=True) @computed_field def data_base64(self) -> str: return b64encode(self.data).decode("ascii") @computed_field def data_hexstr(self) -> str: return self.data.hex().upper()
[docs] @field_validator("data") @classmethod def check_data(cls, data: bytes) -> bytes: """Check for data Parameters ---------- data : bytes Input Returns ------- bytes Valid input Raises ------ ValueError Byte length is not in range between 1 and 80 """ if not (1 <= len(data) <= 80): raise ValueError("Payload length should be in range between 1 and 80.") return data
[docs] @final class PacketParser(common.PacketParserBase): """Packet parser for App_Uart (Mode A)"""
[docs] @staticmethod @override def is_valid(bare_packet: common.BarePacket) -> bool: """Check the given bare packet is valid or not Parameters ---------- bare_packet : common.BarePacket Bare packet content Returns ------- bool True if valid Notes ----- Static overridden method """ if ( (0x00 <= bare_packet.u8_at(0) <= 0x64 or bare_packet.u8_at(0) == 0x78) and bare_packet.u8_at(1) < 0x80 and 3 <= len(bare_packet.payload) <= 82 ): return True return False
[docs] @staticmethod @override def parse(bare_packet: common.BarePacket) -> ParsedPacket | None: """Try to parse the given bare packet Parameters ---------- bare_packet : common.BarePacket Bare packet content Returns ------- ParsedPacket | None Parsed packet data if valid else None Notes ----- Static overridden method """ if not PacketParser.is_valid(bare_packet): return None parsed_packet_data: dict[str, Any] = { "time_parsed": datetime.now(common.Timezone), "packet_type": common.PacketType.APP_UART_ASCII, "sequence_number": None, "source_serial_id": common.UInt32(0), "source_logical_id": bare_packet.u8_at(0), "lqi": None, "supply_voltage": None, "command_id": bare_packet.u8_at(1), "data": bare_packet.payload[2:], } return ParsedPacket(**parsed_packet_data)