2023-10-04 22:25:15 +02:00
|
|
|
import re
|
2023-10-06 02:46:43 +02:00
|
|
|
from dataclasses import dataclass, field
|
2022-04-15 20:02:42 +02:00
|
|
|
from enum import IntEnum, unique
|
|
|
|
from typing import TypeVar
|
2022-04-04 14:48:43 +02:00
|
|
|
|
2023-10-04 15:42:03 +02:00
|
|
|
import channels
|
|
|
|
from asgiref.sync import async_to_sync
|
|
|
|
|
2023-10-06 02:46:43 +02:00
|
|
|
from c3nav.mesh.baseformats import BoolFormat, FixedStrFormat, SimpleFormat, StructType, VarArrayFormat, VarStrFormat
|
|
|
|
from c3nav.mesh.dataformats import (FirmwareAppDescription, LedConfig, MacAddressesListFormat, MacAddressFormat,
|
|
|
|
RangeItemType)
|
|
|
|
from c3nav.mesh.utils import get_mesh_comm_group
|
2022-04-04 14:48:43 +02:00
|
|
|
|
2023-10-05 05:02:01 +02:00
|
|
|
MESH_ROOT_ADDRESS = '00:00:00:00:00:00'
|
2023-10-06 02:39:19 +02:00
|
|
|
MESH_NONE_ADDRESS = '00:00:00:00:00:00'
|
2023-10-05 05:02:01 +02:00
|
|
|
MESH_PARENT_ADDRESS = '00:00:00:ff:ff:ff'
|
2023-10-06 02:39:19 +02:00
|
|
|
MESH_CHILDREN_ADDRESS = '00:00:00:00:ff:ff'
|
2023-10-05 05:02:01 +02:00
|
|
|
MESH_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
|
|
|
|
2023-10-06 02:46:43 +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
|
2023-10-05 02:34:42 +02:00
|
|
|
MESH_ROUTE_REQUEST = 0x07
|
|
|
|
MESH_ROUTE_RESPONSE = 0x08
|
|
|
|
MESH_ROUTE_TRACE = 0x09
|
2023-10-06 02:10:17 +02:00
|
|
|
MESH_ROUTING_FAILED = 0x0a
|
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-05 19:42:36 +02:00
|
|
|
LOCATE_REPORT_RANGE = 0x20
|
|
|
|
|
2022-04-06 23:44:24 +02:00
|
|
|
|
2023-10-04 22:25:15 +02:00
|
|
|
M = TypeVar('M', bound='MeshMessage')
|
2022-04-04 14:48:43 +02:00
|
|
|
|
|
|
|
|
2023-10-03 17:23:29 +02:00
|
|
|
@unique
|
|
|
|
class ChipType(IntEnum):
|
|
|
|
ESP32_S2 = 2
|
|
|
|
ESP32_C3 = 5
|
|
|
|
|
|
|
|
|
2022-04-04 14:48:43 +02:00
|
|
|
@dataclass
|
2023-10-05 19:42:36 +02:00
|
|
|
class MeshMessage(StructType, union_type_field="msg_id"):
|
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)
|
|
|
|
c_structs = {}
|
|
|
|
c_struct_name = None
|
2022-04-04 14:48:43 +02:00
|
|
|
|
|
|
|
# noinspection PyMethodOverriding
|
2023-10-05 19:42:36 +02:00
|
|
|
def __init_subclass__(cls, /, c_struct_name=None, **kwargs):
|
2022-04-04 14:48:43 +02:00
|
|
|
super().__init_subclass__(**kwargs)
|
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
|
2022-04-04 14:48:43 +02:00
|
|
|
|
2023-10-06 02:15:51 +02:00
|
|
|
def send(self, sender=None, exclude_uplink_address=None):
|
2023-10-04 15:42:03 +02:00
|
|
|
async_to_sync(channels.layers.get_channel_layer().group_send)(get_mesh_comm_group(self.dst), {
|
|
|
|
"type": "mesh.send",
|
2023-10-05 04:05:29 +02:00
|
|
|
"sender": sender,
|
2023-10-06 02:15:51 +02:00
|
|
|
"exclude_uplink_address": exclude_uplink_address,
|
2023-10-06 01:06:30 +02:00
|
|
|
"msg": MeshMessage.tojson(self),
|
2023-10-04 15:42:03 +02:00
|
|
|
})
|
|
|
|
|
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
|
2023-10-05 20:55:36 +02:00
|
|
|
def get_variable_name(cls, base_name):
|
|
|
|
return cls.c_struct_name or base_name
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def get_struct_name(cls, base_name):
|
|
|
|
return "mesh_msg_%s_t" % base_name
|
2023-10-04 22:25:15 +02:00
|
|
|
|
|
|
|
@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
|
|
|
|
|
2022-04-04 14:48:43 +02:00
|
|
|
|
|
|
|
@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",
|
|
|
|
})
|
2022-04-04 14:48:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
@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
|
2022-04-04 14:48:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
@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 """
|
2022-04-06 21:05:52 +02:00
|
|
|
pass
|
2022-04-04 14:48:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
@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",
|
|
|
|
})
|
2022-04-04 14:48:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
2023-10-04 22:25:15 +02:00
|
|
|
class BaseDestinationsMessage(MeshMessage, c_struct_name="destinations"):
|
|
|
|
""" downstream node announces served/no longer served destination """
|
2023-10-05 02:34:42 +02:00
|
|
|
addresses: list[str] = field(default_factory=list, metadata={
|
2023-10-04 22:25:15 +02:00
|
|
|
"format": MacAddressesListFormat(),
|
2023-10-05 02:34:42 +02:00
|
|
|
"doc": "adresses of the destinations",
|
|
|
|
"c_name": "addresses",
|
2023-10-04 22:25:15 +02:00
|
|
|
})
|
2022-04-04 14:48:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
@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
|
|
|
|
|
|
|
|
2023-10-05 02:34:42 +02:00
|
|
|
@dataclass
|
|
|
|
class MeshRouteRequestMessage(MeshMessage, msg_id=MeshMessageType.MESH_ROUTE_REQUEST):
|
|
|
|
""" request routing information for node """
|
|
|
|
request_id: int = field(metadata={"format": SimpleFormat('I')})
|
|
|
|
address: str = field(metadata={
|
|
|
|
"format": MacAddressFormat(),
|
|
|
|
"doc": "target address for the route"
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
class MeshRouteResponseMessage(MeshMessage, msg_id=MeshMessageType.MESH_ROUTE_RESPONSE):
|
|
|
|
""" reporting the routing table entry to the given address """
|
|
|
|
request_id: int = field(metadata={"format": SimpleFormat('I')})
|
|
|
|
route: str = field(metadata={
|
|
|
|
"format": MacAddressFormat(),
|
|
|
|
"doc": "routing table entry or 00:00:00:00:00:00"
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
class MeshRouteTraceMessage(MeshMessage, msg_id=MeshMessageType.MESH_ROUTE_TRACE):
|
|
|
|
""" special message, collects all hop adresses on its way """
|
|
|
|
request_id: int = field(metadata={"format": SimpleFormat('I')})
|
|
|
|
trace: list[str] = field(default_factory=list, metadata={
|
|
|
|
"format": MacAddressesListFormat(),
|
|
|
|
"doc": "addresses encountered by this message",
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2023-10-06 02:10:17 +02:00
|
|
|
@dataclass
|
|
|
|
class MeshRoutingFailedMessage(MeshMessage, msg_id=MeshMessageType.MESH_ROUTING_FAILED):
|
|
|
|
""" TODO description"""
|
|
|
|
address: str = field(metadata={"format": MacAddressFormat()})
|
|
|
|
|
|
|
|
|
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')})
|
2023-10-06 02:10:17 +02:00
|
|
|
app_desc: FirmwareAppDescription = field(metadata={'json_embed': True})
|
2022-04-06 23:44:24 +02:00
|
|
|
|
2023-10-04 23:05:36 +02:00
|
|
|
@classmethod
|
|
|
|
def upgrade_json(cls, data):
|
|
|
|
if 'revision' in data:
|
|
|
|
data['revision_major'], data['revision_minor'] = data.pop('revision')
|
|
|
|
return data
|
2023-10-02 22:02:25 +02:00
|
|
|
|
2023-10-03 17:23:29 +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 """
|
2023-10-06 02:10:17 +02:00
|
|
|
led_config: LedConfig = field(metadata={"c_embed": True})
|
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')})
|
2023-10-05 19:42:36 +02:00
|
|
|
|
|
|
|
|
|
|
|
@dataclass
|
|
|
|
class LocateReportRangeMessage(MeshMessage, msg_id=MeshMessageType.LOCATE_REPORT_RANGE):
|
|
|
|
""" report distance to given nodes """
|
2023-10-06 02:46:43 +02:00
|
|
|
ranges: dict[str, int] = field(metadata={"format": VarArrayFormat(RangeItemType)})
|