mirror of
https://github.com/meshtastic/python.git
synced 2025-12-24 16:37:51 -05:00
188 lines
7.3 KiB
Python
188 lines
7.3 KiB
Python
"""Stream Interface base class
|
|
"""
|
|
import logging
|
|
import threading
|
|
import time
|
|
import traceback
|
|
import serial
|
|
|
|
|
|
from meshtastic.mesh_interface import MeshInterface
|
|
from meshtastic.util import stripnl, is_windows11
|
|
|
|
|
|
START1 = 0x94
|
|
START2 = 0xc3
|
|
HEADER_LEN = 4
|
|
MAX_TO_FROM_RADIO_SIZE = 512
|
|
|
|
|
|
class StreamInterface(MeshInterface):
|
|
"""Interface class for meshtastic devices over a stream link (serial, TCP, etc)"""
|
|
|
|
def __init__(self, debugOut=None, noProto=False, connectNow=True):
|
|
"""Constructor, opens a connection to self.stream
|
|
|
|
Keyword Arguments:
|
|
debugOut {stream} -- If a stream is provided, any debug serial output from the
|
|
device will be emitted to that stream. (default: {None})
|
|
|
|
Raises:
|
|
Exception: [description]
|
|
Exception: [description]
|
|
"""
|
|
|
|
if not hasattr(self, 'stream') and not noProto:
|
|
raise Exception(
|
|
"StreamInterface is now abstract (to update existing code create SerialInterface instead)")
|
|
self._rxBuf = bytes() # empty
|
|
self._wantExit = False
|
|
|
|
self.is_windows11 = is_windows11()
|
|
|
|
# FIXME, figure out why daemon=True causes reader thread to exit too early
|
|
self._rxThread = threading.Thread(target=self.__reader, args=(), daemon=True)
|
|
|
|
MeshInterface.__init__(self, debugOut=debugOut, noProto=noProto)
|
|
|
|
# Start the reader thread after superclass constructor completes init
|
|
if connectNow:
|
|
self.connect()
|
|
if not noProto:
|
|
self.waitForConfig()
|
|
|
|
def connect(self):
|
|
"""Connect to our radio
|
|
|
|
Normally this is called automatically by the constructor, but if you
|
|
passed in connectNow=False you can manually start the reading thread later.
|
|
"""
|
|
|
|
# Send some bogus UART characters to force a sleeping device to wake, and
|
|
# if the reading statemachine was parsing a bad packet make sure
|
|
# we write enought start bytes to force it to resync (we don't use START1
|
|
# because we want to ensure it is looking for START1)
|
|
p = bytearray([START2] * 32)
|
|
self._writeBytes(p)
|
|
time.sleep(0.1) # wait 100ms to give device time to start running
|
|
|
|
self._rxThread.start()
|
|
|
|
self._startConfig()
|
|
|
|
if not self.noProto: # Wait for the db download if using the protocol
|
|
self._waitConnected()
|
|
|
|
def _disconnected(self):
|
|
"""We override the superclass implementation to close our port"""
|
|
MeshInterface._disconnected(self)
|
|
|
|
logging.debug("Closing our port")
|
|
# pylint: disable=E0203
|
|
if not self.stream is None:
|
|
# pylint: disable=E0203
|
|
self.stream.close()
|
|
# pylint: disable=W0201
|
|
self.stream = None
|
|
|
|
def _writeBytes(self, b):
|
|
"""Write an array of bytes to our stream and flush"""
|
|
if self.stream: # ignore writes when stream is closed
|
|
self.stream.write(b)
|
|
self.stream.flush()
|
|
# win11 might need a bit more time, too
|
|
if self.is_windows11:
|
|
time.sleep(1.0)
|
|
else:
|
|
# we sleep here to give the TBeam a chance to work
|
|
time.sleep(0.1)
|
|
|
|
def _readBytes(self, length):
|
|
"""Read an array of bytes from our stream"""
|
|
if self.stream:
|
|
return self.stream.read(length)
|
|
else:
|
|
return None
|
|
|
|
def _sendToRadioImpl(self, toRadio):
|
|
"""Send a ToRadio protobuf to the device"""
|
|
logging.debug(f"Sending: {stripnl(toRadio)}")
|
|
b = toRadio.SerializeToString()
|
|
bufLen = len(b)
|
|
# We convert into a string, because the TCP code doesn't work with byte arrays
|
|
header = bytes([START1, START2, (bufLen >> 8) & 0xff, bufLen & 0xff])
|
|
logging.debug(f'sending header:{header} b:{b}')
|
|
self._writeBytes(header + b)
|
|
|
|
def close(self):
|
|
"""Close a connection to the device"""
|
|
logging.debug("Closing stream")
|
|
MeshInterface.close(self)
|
|
# pyserial cancel_read doesn't seem to work, therefore we ask the
|
|
# reader thread to close things for us
|
|
self._wantExit = True
|
|
if self._rxThread != threading.current_thread():
|
|
self._rxThread.join() # wait for it to exit
|
|
|
|
def __reader(self):
|
|
"""The reader thread that reads bytes from our stream"""
|
|
logging.debug('in __reader()')
|
|
empty = bytes()
|
|
|
|
try:
|
|
while not self._wantExit:
|
|
#logging.debug("reading character")
|
|
b = self._readBytes(1)
|
|
#logging.debug("In reader loop")
|
|
#logging.debug(f"read returned {b}")
|
|
if len(b) > 0:
|
|
c = b[0]
|
|
#logging.debug(f'c:{c}')
|
|
ptr = len(self._rxBuf)
|
|
|
|
# Assume we want to append this byte, fixme use bytearray instead
|
|
self._rxBuf = self._rxBuf + b
|
|
|
|
if ptr == 0: # looking for START1
|
|
if c != START1:
|
|
self._rxBuf = empty # failed to find start
|
|
if self.debugOut is not None:
|
|
try:
|
|
self.debugOut.write(b.decode("utf-8"))
|
|
except:
|
|
self.debugOut.write('?')
|
|
|
|
elif ptr == 1: # looking for START2
|
|
if c != START2:
|
|
self._rxBuf = empty # failed to find start2
|
|
elif ptr >= HEADER_LEN - 1: # we've at least got a header
|
|
#logging.debug('at least we received a header')
|
|
# big endian length follows header
|
|
packetlen = (self._rxBuf[2] << 8) + self._rxBuf[3]
|
|
|
|
if ptr == HEADER_LEN - 1: # we _just_ finished reading the header, validate length
|
|
if packetlen > MAX_TO_FROM_RADIO_SIZE:
|
|
self._rxBuf = empty # length was out out bounds, restart
|
|
|
|
if len(self._rxBuf) != 0 and ptr + 1 >= packetlen + HEADER_LEN:
|
|
try:
|
|
self._handleFromRadio(self._rxBuf[HEADER_LEN:])
|
|
except Exception as ex:
|
|
logging.error(f"Error while handling message from radio {ex}")
|
|
traceback.print_exc()
|
|
self._rxBuf = empty
|
|
else:
|
|
# logging.debug(f"timeout")
|
|
pass
|
|
except serial.SerialException as ex:
|
|
if not self._wantExit: # We might intentionally get an exception during shutdown
|
|
logging.warning(f"Meshtastic serial port disconnected, disconnecting... {ex}")
|
|
except OSError as ex:
|
|
if not self._wantExit: # We might intentionally get an exception during shutdown
|
|
logging.error(f"Unexpected OSError, terminating meshtastic reader... {ex}")
|
|
except Exception as ex:
|
|
logging.error(f"Unexpected exception, terminating meshtastic reader... {ex}")
|
|
finally:
|
|
logging.debug("reader is exiting")
|
|
self._disconnected()
|