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

352 lines
12 KiB
Python
Raw Normal View History

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
import channels
2023-11-07 16:35:46 +01:00
from channels.db import database_sync_to_async
2023-10-20 16:22:32 +02:00
from c3nav.mesh.baseformats import (BoolFormat, EnumFormat, FixedStrFormat, SimpleFormat, StructType, VarArrayFormat,
VarBytesFormat, VarStrFormat, normalize_name)
from c3nav.mesh.dataformats import (BoardConfig, ChipType, FirmwareAppDescription, MacAddressesListFormat,
MacAddressFormat, RangeResultItem, RawFTMEntry)
2023-11-07 16:35:46 +01:00
from c3nav.mesh.utils import MESH_ALL_UPLINKS_GROUP
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_HARDWARE = 0x11
CONFIG_BOARD = 0x12
CONFIG_FIRMWARE = 0x13
2022-04-15 20:02:42 +02:00
CONFIG_UPLINK = 0x14
CONFIG_POSITION = 0x15
2022-04-06 23:44:24 +02:00
2023-10-20 16:22:32 +02:00
OTA_STATUS = 0x20
OTA_REQUEST_STATUS = 0x21
OTA_START = 0x22
OTA_URL = 0x23
OTA_FRAGMENT = 0x24
2023-11-23 19:03:20 +01:00
OTA_REQUEST_FRAGMENTS = 0x25
2023-10-20 16:22:32 +02:00
OTA_APPLY = 0x26
OTA_REBOOT = 0x27
LOCATE_REQUEST_RANGE = 0x30
LOCATE_RANGE_RESULTS = 0x31
2023-10-26 14:22:52 +02:00
LOCATE_RAW_FTM_RESULTS = 0x32
2023-10-20 15:04:36 +02:00
@property
def pretty_name(self):
2023-10-20 15:41:46 +02:00
name = self.name.replace('_', ' ').lower()
if name.startswith('config'):
name = name.removeprefix('config').strip()+' config'
name.replace('ota', 'OTA')
return name
2023-10-20 15:04:36 +02:00
2022-04-06 23:44:24 +02:00
2023-10-04 22:25:15 +02:00
M = TypeVar('M', bound='MeshMessage')
@dataclass
2023-10-20 15:41:46 +02:00
class MeshMessage(StructType, union_type_field="msg_type"):
2023-10-04 22:25:15 +02:00
dst: str = field(metadata={"format": MacAddressFormat()})
src: str = field(metadata={"format": MacAddressFormat()})
2023-10-20 16:22:32 +02:00
msg_type: MeshMessageType = field(metadata={"format": EnumFormat('B', c_definition=False)}, init=False, repr=False)
2023-10-04 22:25:15 +02:00
c_structs = {}
c_struct_name = None
# noinspection PyMethodOverriding
def __init_subclass__(cls, /, c_struct_name=None, **kwargs):
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
2023-11-07 16:35:46 +01:00
async def send(self, sender=None, exclude_uplink_address=None) -> bool:
data = {
"type": "mesh.send",
2023-10-05 04:05:29 +02:00
"sender": sender,
"exclude_uplink_address": exclude_uplink_address,
2023-10-06 01:06:30 +02:00
"msg": MeshMessage.tojson(self),
2023-11-07 16:35:46 +01:00
}
if self.dst in (MESH_CHILDREN_ADDRESS, MESH_BROADCAST_ADDRESS):
await channels.layers.get_channel_layer().group_send(MESH_ALL_UPLINKS_GROUP, data)
return True
from c3nav.mesh.models import MeshNode
uplink = database_sync_to_async(MeshNode.get_node_and_uplink)(self.dst)
if not uplink:
return False
if uplink.node_id == exclude_uplink_address:
return False
await channels.layers.get_channel_layer().send(uplink.name, data)
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
2023-10-04 22:25:15 +02:00
@classmethod
def get_c_enum_name(cls):
return normalize_name(cls.__name__.removeprefix('Mesh').removesuffix('Message')).upper()
2023-10-04 22:25:15 +02:00
@dataclass
2023-10-20 15:41:46 +02:00
class NoopMessage(MeshMessage, msg_type=MeshMessageType.NOOP):
2023-10-04 22:25:15 +02:00
""" 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-20 15:41:46 +02:00
class EchoRequestMessage(BaseEchoMessage, msg_type=MeshMessageType.ECHO_REQUEST):
2023-10-04 22:25:15 +02:00
""" repeat back string """
pass
@dataclass
2023-10-20 15:41:46 +02:00
class EchoResponseMessage(BaseEchoMessage, msg_type=MeshMessageType.ECHO_RESPONSE):
2023-10-04 22:25:15 +02:00
""" repeat back string """
pass
@dataclass
2023-10-20 15:41:46 +02:00
class MeshSigninMessage(MeshMessage, msg_type=MeshMessageType.MESH_SIGNIN):
2023-10-04 22:25:15 +02:00
""" node says hello to upstream node """
pass
@dataclass
2023-10-20 15:41:46 +02:00
class MeshLayerAnnounceMessage(MeshMessage, msg_type=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 """
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
})
@dataclass
2023-10-20 15:41:46 +02:00
class MeshAddDestinationsMessage(BaseDestinationsMessage, msg_type=MeshMessageType.MESH_ADD_DESTINATIONS):
2023-10-04 22:25:15 +02:00
""" downstream node announces served destination """
pass
@dataclass
2023-10-20 15:41:46 +02:00
class MeshRemoveDestinationsMessage(BaseDestinationsMessage, msg_type=MeshMessageType.MESH_REMOVE_DESTINATIONS):
2023-10-04 22:25:15 +02:00
""" 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
2023-10-20 15:41:46 +02:00
class MeshRouteRequestMessage(MeshMessage, msg_type=MeshMessageType.MESH_ROUTE_REQUEST):
2023-10-05 02:34:42 +02:00
""" 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
2023-10-20 15:41:46 +02:00
class MeshRouteResponseMessage(MeshMessage, msg_type=MeshMessageType.MESH_ROUTE_RESPONSE):
2023-10-05 02:34:42 +02:00
""" 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
2023-10-20 15:41:46 +02:00
class MeshRouteTraceMessage(MeshMessage, msg_type=MeshMessageType.MESH_ROUTE_TRACE):
2023-10-05 02:34:42 +02:00
""" 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
2023-10-20 15:41:46 +02:00
class MeshRoutingFailedMessage(MeshMessage, msg_type=MeshMessageType.MESH_ROUTING_FAILED):
2023-10-06 02:10:17 +02:00
""" TODO description"""
address: str = field(metadata={"format": MacAddressFormat()})
2022-04-06 22:56:08 +02:00
@dataclass
2023-10-20 15:41:46 +02:00
class ConfigDumpMessage(MeshMessage, msg_type=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-20 15:41:46 +02:00
class ConfigHardwareMessage(MeshMessage, msg_type=MeshMessageType.CONFIG_HARDWARE):
""" respond hardware/chip info """
chip: ChipType = field(metadata={
2023-10-20 16:22:32 +02:00
"format": EnumFormat("H", c_definition=False),
2023-10-04 22:25:15 +02:00
"c_name": "chip_id",
})
revision_major: int = field(metadata={"format": SimpleFormat('B')})
revision_minor: int = field(metadata={"format": SimpleFormat('B')})
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
@dataclass
2023-10-20 15:41:46 +02:00
class ConfigBoardMessage(MeshMessage, msg_type=MeshMessageType.CONFIG_BOARD):
""" set/respond board config """
board_config: BoardConfig = field(metadata={"c_embed": True, "json_embed": True})
@dataclass
2023-10-20 15:41:46 +02:00
class ConfigFirmwareMessage(MeshMessage, msg_type=MeshMessageType.CONFIG_FIRMWARE):
""" respond firmware info """
app_desc: FirmwareAppDescription = field(metadata={'json_embed': True})
2023-10-04 22:25:15 +02:00
2022-04-06 23:44:24 +02:00
@dataclass
2023-10-20 15:41:46 +02:00
class ConfigPositionMessage(MeshMessage, msg_type=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-20 15:41:46 +02:00
class ConfigUplinkMessage(MeshMessage, msg_type=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-20 16:22:32 +02:00
@dataclass
class OTAStatusMessage(MeshMessage, msg_type=MeshMessageType.OTA_STATUS):
""" report OTA status """
update_id: int = field(metadata={"format": SimpleFormat('I')})
received_bytes: int = field(metadata={"format": SimpleFormat('I')})
2023-11-23 18:56:33 +01:00
auto_apply: bool = field(metadata={"format": BoolFormat()})
2023-10-20 16:22:32 +02:00
app_desc: FirmwareAppDescription = field()
@dataclass
class OTARequestStatusMessage(MeshMessage, msg_type=MeshMessageType.OTA_REQUEST_STATUS):
""" request OTA status """
@dataclass
class OTAStartMessage(MeshMessage, msg_type=MeshMessageType.OTA_START):
""" instruct node to start OTA """
update_id: int = field(metadata={"format": SimpleFormat('I')})
total_bytes: int = field(metadata={"format": SimpleFormat('I')})
@dataclass
class OTAURLMessage(MeshMessage, msg_type=MeshMessageType.OTA_URL):
""" supply download URL for OTA update and who to distribute it to """
update_id: int = field(metadata={"format": SimpleFormat('I')})
distribute_to: str = field(metadata={"format": MacAddressFormat()})
url: str = field(metadata={"format": VarStrFormat()})
@dataclass
class OTAFragmentMessage(MeshMessage, msg_type=MeshMessageType.OTA_FRAGMENT):
""" supply OTA fragment """
update_id: int = field(metadata={"format": SimpleFormat('I')})
2023-11-23 18:56:33 +01:00
chunk: int = field(metadata={"format": SimpleFormat('H')})
2023-10-20 16:22:32 +02:00
data: str = field(metadata={"format": VarBytesFormat()})
@dataclass
2023-11-23 19:03:20 +01:00
class OTARequestFragmentsMessage(MeshMessage, msg_type=MeshMessageType.OTA_REQUEST_FRAGMENTS):
2023-10-20 23:18:15 +02:00
""" request fragment after we haven't gotten one for a while """
2023-10-20 16:22:32 +02:00
update_id: int = field(metadata={"format": SimpleFormat('I')})
2023-11-23 19:03:20 +01:00
chunks: list[int] = field(metadata={"format": VarArrayFormat(SimpleFormat('H'))})
2023-10-20 16:22:32 +02:00
@dataclass
class OTAApplyMessage(MeshMessage, msg_type=MeshMessageType.OTA_APPLY):
""" apply OTA, optionally instruct to apply it when done """
update_id: int = field(metadata={"format": SimpleFormat('I')})
when_done: bool = field(metadata={"format": BoolFormat()})
@dataclass
class OTARebootMessage(MeshMessage, msg_type=MeshMessageType.OTA_REBOOT):
""" announcing OTA reboot """
update_id: int = field(metadata={"format": SimpleFormat('I')})
@dataclass
2023-10-20 15:41:46 +02:00
class LocateRequestRangeMessage(MeshMessage, msg_type=MeshMessageType.LOCATE_REQUEST_RANGE):
2023-10-16 22:24:14 +02:00
""" request to report distance to all nearby nodes """
pass
@dataclass
2023-10-20 15:41:46 +02:00
class LocateRangeResults(MeshMessage, msg_type=MeshMessageType.LOCATE_RANGE_RESULTS):
2023-10-16 22:24:14 +02:00
""" reports distance to given nodes """
2023-10-26 14:22:52 +02:00
ranges: list[RangeResultItem] = field(metadata={"format": VarArrayFormat(RangeResultItem)})
@dataclass
class LocateRawFTMResults(MeshMessage, msg_type=MeshMessageType.LOCATE_RAW_FTM_RESULTS):
""" reports distance to given nodes """
2023-10-28 19:56:03 +02:00
peer: str = field(metadata={"format": MacAddressFormat()})
2023-10-26 14:22:52 +02:00
results: list[RawFTMEntry] = field(metadata={"format": VarArrayFormat(RawFTMEntry)})