mirror of
https://github.com/jkingsman/Remote-Terminal-for-MeshCore.git
synced 2026-03-28 17:43:05 +01:00
202 lines
6.9 KiB
Python
202 lines
6.9 KiB
Python
import logging
|
|
import time
|
|
from typing import TYPE_CHECKING
|
|
|
|
from meshcore import EventType
|
|
|
|
from app.models import Contact
|
|
from app.packet_processor import process_raw_packet, track_pending_repeat
|
|
from app.repository import ContactRepository, MessageRepository
|
|
from app.websocket import broadcast_event
|
|
|
|
if TYPE_CHECKING:
|
|
from meshcore.events import Event
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
# Track pending ACKs: expected_ack_code -> (message_id, timestamp, timeout_ms)
|
|
_pending_acks: dict[str, tuple[int, float, int]] = {}
|
|
|
|
|
|
def track_pending_ack(expected_ack: str, message_id: int, timeout_ms: int) -> None:
|
|
"""Track a pending ACK for a direct message."""
|
|
_pending_acks[expected_ack] = (message_id, time.time(), timeout_ms)
|
|
logger.debug("Tracking pending ACK %s for message %d (timeout %dms)", expected_ack, message_id, timeout_ms)
|
|
|
|
|
|
def _cleanup_expired_acks() -> None:
|
|
"""Remove expired pending ACKs."""
|
|
now = time.time()
|
|
expired = []
|
|
for code, (msg_id, created_at, timeout_ms) in _pending_acks.items():
|
|
if now - created_at > (timeout_ms / 1000) * 2: # 2x timeout as buffer
|
|
expired.append(code)
|
|
for code in expired:
|
|
del _pending_acks[code]
|
|
logger.debug("Expired pending ACK %s", code)
|
|
|
|
|
|
async def on_contact_message(event: "Event") -> None:
|
|
"""Handle incoming direct messages.
|
|
|
|
Direct messages are decrypted by MeshCore library using ECDH key exchange.
|
|
The packet processor cannot decrypt these without the node's private key.
|
|
"""
|
|
payload = event.payload
|
|
|
|
# Skip CLI command responses (txt_type=1) - these are handled by the command endpoint
|
|
# and should not be stored in the database or broadcast via WebSocket
|
|
txt_type = payload.get("txt_type", 0)
|
|
if txt_type == 1:
|
|
logger.debug("Skipping CLI response from %s (txt_type=1)", payload.get("pubkey_prefix"))
|
|
return
|
|
|
|
logger.debug("Received direct message from %s", payload.get("pubkey_prefix"))
|
|
|
|
# Get full public key if available, otherwise use prefix
|
|
sender_pubkey = payload.get("public_key") or payload.get("pubkey_prefix", "")
|
|
received_at = int(time.time())
|
|
|
|
# Look up full public key from contact database if we only have prefix
|
|
if len(sender_pubkey) < 64:
|
|
contact = await ContactRepository.get_by_key_prefix(sender_pubkey)
|
|
if contact:
|
|
sender_pubkey = contact.public_key
|
|
|
|
# Try to create message - INSERT OR IGNORE handles duplicates atomically
|
|
msg_id = await MessageRepository.create(
|
|
msg_type="PRIV",
|
|
text=payload.get("text", ""),
|
|
conversation_key=sender_pubkey,
|
|
sender_timestamp=payload.get("sender_timestamp"),
|
|
received_at=received_at,
|
|
path_len=payload.get("path_len"),
|
|
txt_type=payload.get("txt_type", 0),
|
|
signature=payload.get("signature"),
|
|
)
|
|
|
|
if msg_id is None:
|
|
# Duplicate message (same content from same sender) - skip broadcast
|
|
logger.debug("Duplicate direct message from %s ignored", sender_pubkey[:12])
|
|
return
|
|
|
|
# Broadcast only genuinely new messages
|
|
broadcast_event("message", {
|
|
"id": msg_id,
|
|
"type": "PRIV",
|
|
"conversation_key": sender_pubkey,
|
|
"text": payload.get("text", ""),
|
|
"sender_timestamp": payload.get("sender_timestamp"),
|
|
"received_at": received_at,
|
|
"path_len": payload.get("path_len"),
|
|
"txt_type": payload.get("txt_type", 0),
|
|
"signature": payload.get("signature"),
|
|
"outgoing": False,
|
|
"acked": False,
|
|
})
|
|
|
|
# Update contact last_seen and last_contacted
|
|
contact = await ContactRepository.get_by_key_prefix(sender_pubkey)
|
|
if contact:
|
|
await ContactRepository.update_last_contacted(contact.public_key, received_at)
|
|
|
|
|
|
async def on_rx_log_data(event: "Event") -> None:
|
|
"""Store raw RF packet data and process via centralized packet processor.
|
|
|
|
This is the unified entry point for all RF packets. The packet processor
|
|
handles channel messages (GROUP_TEXT) and advertisements (ADVERT).
|
|
"""
|
|
payload = event.payload
|
|
logger.debug("Received RX log data packet")
|
|
|
|
if "payload" not in payload:
|
|
logger.warning("RX_LOG_DATA event missing 'payload' field")
|
|
return
|
|
|
|
raw_hex = payload["payload"]
|
|
raw_bytes = bytes.fromhex(raw_hex)
|
|
|
|
await process_raw_packet(
|
|
raw_bytes=raw_bytes,
|
|
snr=payload.get("snr"),
|
|
rssi=payload.get("rssi"),
|
|
)
|
|
|
|
|
|
async def on_path_update(event: "Event") -> None:
|
|
"""Handle path update events."""
|
|
payload = event.payload
|
|
logger.debug("Path update for %s", payload.get("pubkey_prefix"))
|
|
|
|
pubkey_prefix = payload.get("pubkey_prefix", "")
|
|
path = payload.get("path", "")
|
|
path_len = payload.get("path_len", -1)
|
|
|
|
existing = await ContactRepository.get_by_key_prefix(pubkey_prefix)
|
|
if existing:
|
|
await ContactRepository.update_path(existing.public_key, path, path_len)
|
|
|
|
|
|
async def on_new_contact(event: "Event") -> None:
|
|
"""Handle new contact from radio's internal contact database.
|
|
|
|
This is different from RF advertisements - these are contacts synced
|
|
from the radio's stored contact list.
|
|
"""
|
|
payload = event.payload
|
|
public_key = payload.get("public_key", "")
|
|
|
|
if not public_key:
|
|
logger.warning("Received new contact event with no public_key, skipping")
|
|
return
|
|
|
|
logger.debug("New contact: %s", public_key[:12])
|
|
|
|
contact_data = {
|
|
**Contact.from_radio_dict(public_key, payload, on_radio=True),
|
|
"last_seen": int(time.time()),
|
|
}
|
|
await ContactRepository.upsert(contact_data)
|
|
|
|
broadcast_event("contact", contact_data)
|
|
|
|
|
|
async def on_ack(event: "Event") -> None:
|
|
"""Handle ACK events for direct messages."""
|
|
payload = event.payload
|
|
ack_code = payload.get("code", "")
|
|
|
|
if not ack_code:
|
|
logger.debug("Received ACK with no code")
|
|
return
|
|
|
|
logger.debug("Received ACK with code %s", ack_code)
|
|
|
|
_cleanup_expired_acks()
|
|
|
|
if ack_code in _pending_acks:
|
|
message_id, _, _ = _pending_acks.pop(ack_code)
|
|
logger.info("ACK received for message %d", message_id)
|
|
|
|
ack_count = await MessageRepository.increment_ack_count(message_id)
|
|
broadcast_event("message_acked", {"message_id": message_id, "ack_count": ack_count})
|
|
else:
|
|
logger.debug("ACK code %s does not match any pending messages", ack_code)
|
|
|
|
|
|
def register_event_handlers(meshcore) -> None:
|
|
"""Register event handlers with the MeshCore instance.
|
|
|
|
Note: CHANNEL_MSG_RECV and ADVERTISEMENT events are NOT subscribed.
|
|
These are handled by the packet processor via RX_LOG_DATA to avoid
|
|
duplicate processing and ensure consistent handling.
|
|
"""
|
|
meshcore.subscribe(EventType.CONTACT_MSG_RECV, on_contact_message)
|
|
meshcore.subscribe(EventType.RX_LOG_DATA, on_rx_log_data)
|
|
meshcore.subscribe(EventType.PATH_UPDATE, on_path_update)
|
|
meshcore.subscribe(EventType.NEW_CONTACT, on_new_contact)
|
|
meshcore.subscribe(EventType.ACK, on_ack)
|
|
logger.info("Event handlers registered")
|