team-3/src/c3nav/mesh/messages.py

312 lines
10 KiB
Python
Raw Normal View History

2023-10-04 22:25:15 +02:00
import re
2022-04-15 20:02:42 +02:00
from dataclasses import asdict, dataclass, field, fields, is_dataclass
from enum import IntEnum, unique
2023-10-04 22:25:15 +02:00
from itertools import chain
2022-04-15 20:02:42 +02:00
from typing import TypeVar
import channels
from asgiref.sync import async_to_sync
2023-10-04 22:25:15 +02:00
from c3nav.mesh.utils import get_mesh_comm_group, indent_c
2022-04-15 20:02:42 +02:00
from c3nav.mesh.dataformats import (BoolFormat, FixedStrFormat, HexFormat, LedConfig, LedConfigFormat,
MacAddressesListFormat, MacAddressFormat, SimpleFormat, VarStrFormat)
2022-04-15 20:02:42 +02:00
ROOT_ADDRESS = '00:00:00:00:00:00'
PARENT_ADDRESS = '00:00:00:ff:ff:ff'
BROADCAST_ADDRESS = 'ff:ff:ff:ff:ff:ff'
2022-04-15 20:02:42 +02:00
NO_LAYER = 0xFF
2022-04-06 23:44:24 +02:00
2022-04-15 20:02:42 +02:00
@unique
2023-10-04 15:44:54 +02:00
class MeshMessageType(IntEnum):
2023-10-04 22:25:15 +02:00
NOOP = 0x00
2022-04-15 20:02:42 +02:00
ECHO_REQUEST = 0x01
ECHO_RESPONSE = 0x02
2022-04-06 23:44:24 +02:00
2022-04-15 20:02:42 +02:00
MESH_SIGNIN = 0x03
MESH_LAYER_ANNOUNCE = 0x04
MESH_ADD_DESTINATIONS = 0x05
MESH_REMOVE_DESTINATIONS = 0x06
2022-04-06 23:44:24 +02:00
2022-04-15 20:02:42 +02:00
CONFIG_DUMP = 0x10
CONFIG_FIRMWARE = 0x11
CONFIG_POSITION = 0x12
CONFIG_LED = 0x13
CONFIG_UPLINK = 0x14
2022-04-06 23:44:24 +02:00
2023-10-04 22:25:15 +02:00
M = TypeVar('M', bound='MeshMessage')
@unique
class ChipType(IntEnum):
ESP32_S2 = 2
ESP32_C3 = 5
@dataclass
2023-10-04 15:44:54 +02:00
class MeshMessage:
2023-10-04 22:25:15 +02:00
dst: str = field(metadata={"format": MacAddressFormat()})
src: str = field(metadata={"format": MacAddressFormat()})
msg_id: int = field(metadata={"format": SimpleFormat('B')}, init=False, repr=False)
msg_types = {}
2023-10-04 22:25:15 +02:00
c_structs = {}
c_struct_name = None
# noinspection PyMethodOverriding
2023-10-04 22:25:15 +02:00
def __init_subclass__(cls, /, msg_id=None, c_struct_name=None, **kwargs):
super().__init_subclass__(**kwargs)
2023-10-04 22:25:15 +02:00
if msg_id is not None:
cls.msg_id = msg_id
2023-10-04 15:44:54 +02:00
if msg_id in MeshMessage.msg_types:
2022-04-15 20:02:42 +02:00
raise TypeError('duplicate use of msg_id %d' % msg_id)
2023-10-04 15:44:54 +02:00
MeshMessage.msg_types[msg_id] = cls
2023-10-04 22:25:15 +02:00
if c_struct_name:
cls.c_struct_name = c_struct_name
if c_struct_name in MeshMessage.c_structs:
raise TypeError('duplicate use of c_struct_name %s' % c_struct_name)
MeshMessage.c_structs[c_struct_name] = cls
def encode(self):
2022-04-06 17:25:46 +02:00
data = bytes()
for field_ in fields(self):
2023-10-04 22:25:15 +02:00
data += field_.metadata["format"].encode(getattr(self, field_.name))
return data
@classmethod
2022-04-15 20:02:42 +02:00
def decode(cls, data: bytes) -> M:
2022-04-06 17:25:46 +02:00
klass = cls.msg_types[data[12]]
values = {}
for field_ in fields(klass):
2023-10-04 22:25:15 +02:00
values[field_.name], data = field_.metadata["format"].decode(data)
2022-04-06 17:25:46 +02:00
values.pop('msg_id')
return klass(**values)
2022-04-15 20:02:42 +02:00
def tojson(self):
return asdict(self)
@classmethod
def fromjson(cls, data) -> M:
kwargs = data.copy()
klass = cls.msg_types[kwargs.pop('msg_id')]
kwargs = klass.upgrade_json(kwargs)
names = set(field.name for field in fields(klass))
2022-04-15 20:02:42 +02:00
for field_ in fields(klass):
if is_dataclass(field_.type):
kwargs[field_.name] = field_.type.fromjson(kwargs[field_.name])
return klass(**kwargs)
@classmethod
def upgrade_json(cls, data):
return data
def send(self):
async_to_sync(channels.layers.get_channel_layer().group_send)(get_mesh_comm_group(self.dst), {
"type": "mesh.send",
"msg": self.tojson()
})
2023-10-04 22:25:15 +02:00
@classmethod
def get_ignore_c_fields(self):
return set()
@classmethod
def get_additional_c_fields(self):
return ()
@classmethod
def get_c_struct(cls):
ignore_fields = cls.get_ignore_c_fields()
if cls != MeshMessage:
ignore_fields |= set(field.name for field in fields(MeshMessage))
items = tuple(
(
tuple(field.metadata["format"].get_c_struct(field.metadata.get("c_name", field.name)).split("\n")),
field.metadata.get("doc", None),
)
for field in fields(cls)
if field.name not in ignore_fields
)
if not items:
return ""
max_line_len = max(len(line) for line in chain(*(code for code, doc in items)))
msg_comment = cls.__doc__.strip()
return "%(comment)stypedef struct __packed {\n%(elements)s\n} %(name)s;" % {
"comment": ("/** %s */\n" % msg_comment) if msg_comment else "",
"elements": indent_c(
"\n".join(chain(*(
(code if not comment
else (code[:-1]+("%s /** %s */" % (code[-1].ljust(max_line_len), comment),)))
for code, comment in items
), cls.get_additional_c_fields()))
),
"name": "mesh_msg_%s_t" % cls.get_c_struct_name(),
}
@classmethod
def get_var_num(cls):
return sum((getattr(field.metadata["format"], "var_num", 0) for field in fields(cls)), start=0)
@classmethod
def get_c_struct_name(cls):
return (
cls.c_struct_name if cls.c_struct_name else
re.sub(
r"([a-z])([A-Z])",
r"\1_\2",
cls.__name__.removeprefix('Mesh').removesuffix('Message')
).lower().replace('config', 'cfg').replace('firmware', 'fw').replace('position', 'pos')
)
@classmethod
def get_c_enum_name(cls):
return re.sub(
r"([a-z])([A-Z])",
r"\1_\2",
cls.__name__.removeprefix('Mesh').removesuffix('Message')
).upper().replace('CONFIG', 'CFG').replace('FIRMWARE', 'FW').replace('POSITION', 'POS')
@dataclass
class NoopMessage(MeshMessage, msg_id=MeshMessageType.NOOP):
""" noop """
pass
@dataclass
2023-10-04 22:25:15 +02:00
class BaseEchoMessage(MeshMessage, c_struct_name="echo"):
""" repeat back string """
content: str = field(default='', metadata={
"format": VarStrFormat(),
"doc": "string to echo",
"c_name": "str",
})
@dataclass
2023-10-04 22:25:15 +02:00
class EchoRequestMessage(BaseEchoMessage, msg_id=MeshMessageType.ECHO_REQUEST):
""" repeat back string """
pass
@dataclass
class EchoResponseMessage(BaseEchoMessage, msg_id=MeshMessageType.ECHO_RESPONSE):
""" repeat back string """
pass
@dataclass
2023-10-04 15:44:54 +02:00
class MeshSigninMessage(MeshMessage, msg_id=MeshMessageType.MESH_SIGNIN):
2023-10-04 22:25:15 +02:00
""" node says hello to upstream node """
pass
@dataclass
2023-10-04 15:44:54 +02:00
class MeshLayerAnnounceMessage(MeshMessage, msg_id=MeshMessageType.MESH_LAYER_ANNOUNCE):
2023-10-04 22:25:15 +02:00
""" upstream node announces layer number """
layer: int = field(metadata={
"format": SimpleFormat('B'),
"doc": "mesh layer that the sending node is on",
})
@dataclass
2023-10-04 22:25:15 +02:00
class BaseDestinationsMessage(MeshMessage, c_struct_name="destinations"):
""" downstream node announces served/no longer served destination """
mac_addresses: list[str] = field(default_factory=list, metadata={
"format": MacAddressesListFormat(),
"doc": "mac adresses of the destinations",
"c_name": "mac",
})
@dataclass
2023-10-04 22:25:15 +02:00
class MeshAddDestinationsMessage(BaseDestinationsMessage, msg_id=MeshMessageType.MESH_ADD_DESTINATIONS):
""" downstream node announces served destination """
pass
@dataclass
class MeshRemoveDestinationsMessage(BaseDestinationsMessage, msg_id=MeshMessageType.MESH_REMOVE_DESTINATIONS):
""" downstream node announces no longer served destination """
pass
2022-04-06 22:56:08 +02:00
@dataclass
2023-10-04 15:44:54 +02:00
class ConfigDumpMessage(MeshMessage, msg_id=MeshMessageType.CONFIG_DUMP):
2023-10-04 22:25:15 +02:00
""" request for the node to dump its config """
2022-04-06 22:56:08 +02:00
pass
@dataclass
2023-10-04 15:44:54 +02:00
class ConfigFirmwareMessage(MeshMessage, msg_id=MeshMessageType.CONFIG_FIRMWARE):
2023-10-04 22:25:15 +02:00
""" respond firmware info """
chip: int = field(metadata={
"format": SimpleFormat('H'),
"c_name": "chip_id",
})
revision_major: int = field(metadata={"format": SimpleFormat('B')})
revision_minor: int = field(metadata={"format": SimpleFormat('B')})
magic_word: int = field(metadata={"format": SimpleFormat('I')}, repr=False)
secure_version: int = field(metadata={"format": SimpleFormat('I')})
reserv1: list[int] = field(metadata={"format": SimpleFormat('2I')}, repr=False)
version: str = field(metadata={"format": FixedStrFormat(32)})
project_name: str = field(metadata={"format": FixedStrFormat(32)})
compile_time: str = field(metadata={"format": FixedStrFormat(16)})
compile_date: str = field(metadata={"format": FixedStrFormat(16)})
idf_version: str = field(metadata={"format": FixedStrFormat(32)})
app_elf_sha256: str = field(metadata={"format": HexFormat(32)})
reserv2: list[int] = field(metadata={"format": SimpleFormat('20I')}, repr=False)
2022-04-06 23:44:24 +02:00
@classmethod
def upgrade_json(cls, data):
data = data.copy() # todo: deepcopy?
if 'revision' in data:
data['revision_major'], data['revision_minor'] = data.pop('revision')
return data
2023-10-02 22:02:25 +02:00
def get_chip_display(self):
return ChipType(self.chip).name.replace('_', '-')
2023-10-04 22:25:15 +02:00
@classmethod
def get_ignore_c_fields(self):
return {
"magic_word", "secure_version", "reserv1", "version", "project_name",
"compile_time", "compile_date", "idf_version", "app_elf_sha256", "reserv2"
}
@classmethod
def get_additional_c_fields(self):
return ("esp_app_desc_t app_desc;", )
2022-04-06 23:44:24 +02:00
@dataclass
2023-10-04 15:44:54 +02:00
class ConfigPositionMessage(MeshMessage, msg_id=MeshMessageType.CONFIG_POSITION):
2023-10-04 22:25:15 +02:00
""" set/respond position config """
x_pos: int = field(metadata={"format": SimpleFormat('i')})
y_pos: int = field(metadata={"format": SimpleFormat('i')})
z_pos: int = field(metadata={"format": SimpleFormat('h')})
2022-04-06 23:44:24 +02:00
@dataclass
2023-10-04 15:44:54 +02:00
class ConfigLedMessage(MeshMessage, msg_id=MeshMessageType.CONFIG_LED):
2023-10-04 22:25:15 +02:00
""" set/respond led config """
led_config: LedConfig = field(metadata={"format": LedConfigFormat()})
2022-04-06 23:44:24 +02:00
@dataclass
2023-10-04 15:44:54 +02:00
class ConfigUplinkMessage(MeshMessage, msg_id=MeshMessageType.CONFIG_UPLINK):
2023-10-04 22:25:15 +02:00
""" set/respond uplink config """
enabled: bool = field(metadata={"format": BoolFormat()})
ssid: str = field(metadata={"format": FixedStrFormat(32)})
password: str = field(metadata={"format": FixedStrFormat(64)})
channel: int = field(metadata={"format": SimpleFormat('B')})
udp: bool = field(metadata={"format": BoolFormat()})
ssl: bool = field(metadata={"format": BoolFormat()})
host: str = field(metadata={"format": FixedStrFormat(64)})
port: int = field(metadata={"format": SimpleFormat('H')})