Polish mrbridge and add ftpserve
This commit is contained in:
79
ftpserve.py
Normal file
79
ftpserve.py
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
import argparse
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import serial
|
||||||
|
|
||||||
|
|
||||||
|
kCrc16Polynomial = 0x1021
|
||||||
|
kCrc16Preset = 0xFFFF
|
||||||
|
|
||||||
|
kDefaultSerialPort = None
|
||||||
|
kDefaultBaudrate = 115200
|
||||||
|
kDefaultChunkSize = 0x20
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def crc16(data):
|
||||||
|
crc = kCrc16Preset
|
||||||
|
for c in data:
|
||||||
|
crc ^= c << 8
|
||||||
|
for j in range(8):
|
||||||
|
mix = crc & 0x8000
|
||||||
|
crc <<= 1
|
||||||
|
if mix:
|
||||||
|
crc ^= kCrc16Polynomial
|
||||||
|
crc &= 0xFFFF
|
||||||
|
return crc
|
||||||
|
|
||||||
|
|
||||||
|
def getserial(port=None, baudrate=None, **kwargs):
|
||||||
|
if port is not None:
|
||||||
|
return serial.Serial(port, baudrate)
|
||||||
|
|
||||||
|
# return first /dev/ttyACMx found
|
||||||
|
for i in range(10):
|
||||||
|
try:
|
||||||
|
return serial.Serial(f"/dev/ttyACM{i}", baudrate)
|
||||||
|
except serial.SerialException:
|
||||||
|
pass
|
||||||
|
raise RuntimeError("No serial device available.")
|
||||||
|
|
||||||
|
|
||||||
|
def ftpserve(file, device, chunksize=0x20):
|
||||||
|
with open(file, "rb") as f:
|
||||||
|
data = f.read()
|
||||||
|
device.reset_input_buffer()
|
||||||
|
logger.info(f"Serving file {file}")
|
||||||
|
for i in range(0, len(data), chunksize):
|
||||||
|
device.write(data[i : i + chunksize])
|
||||||
|
r = device.read()
|
||||||
|
if r != b"\x42":
|
||||||
|
raise RuntimeError(f"wrong response {r}")
|
||||||
|
logger.info(f"ack for chunk @{i}")
|
||||||
|
logger.info(f"Done serving file {file}")
|
||||||
|
crc = crc16(data)
|
||||||
|
logger.info(f"File length: {len(data):04x}, crc16: {crc:04x}")
|
||||||
|
|
||||||
|
|
||||||
|
def parse_args():
|
||||||
|
parser = argparse.ArgumentParser(description="Serve a file to ftpget.")
|
||||||
|
parser.add_argument("file", help="file to serve")
|
||||||
|
parser.add_argument("--port", help="path to the serial port")
|
||||||
|
parser.add_argument("--baudrate", default=kDefaultBaudrate)
|
||||||
|
parser.add_argument("--chunksize", default=kDefaultChunkSize)
|
||||||
|
return parser.parse_args()
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
logging.basicConfig(level=logging.INFO)
|
||||||
|
args = parse_args()
|
||||||
|
device = getserial(**vars(args))
|
||||||
|
ftpserve(args.file, device, args.chunksize)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
try:
|
||||||
|
main()
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
pass
|
40
mrbridge.py
40
mrbridge.py
@@ -4,12 +4,29 @@ import threading
|
|||||||
import time
|
import time
|
||||||
import websocket
|
import websocket
|
||||||
|
|
||||||
|
kDefaultSerialPort = None
|
||||||
|
kDefaultBaudrate = 115200
|
||||||
|
|
||||||
|
|
||||||
|
def getserial(port=None, baudrate=None, **kwargs):
|
||||||
|
if port is not None:
|
||||||
|
return serial.Serial(port, baudrate)
|
||||||
|
|
||||||
|
# return first /dev/ttyACMx found
|
||||||
|
for i in range(10):
|
||||||
|
try:
|
||||||
|
return serial.Serial(f"/dev/ttyACM{i}", baudrate)
|
||||||
|
except serial.SerialException:
|
||||||
|
pass
|
||||||
|
raise RuntimeError("No serial device available.")
|
||||||
|
|
||||||
|
|
||||||
def parse_args():
|
def parse_args():
|
||||||
parser = argparse.ArgumentParser(description="Bridge a serial port to a websocket")
|
parser = argparse.ArgumentParser(description="Bridge a serial port to a websocket")
|
||||||
parser.add_argument("port", help="path to the serial port")
|
parser.add_argument(
|
||||||
parser.add_argument("--baudrate", default=115200)
|
"--port", help="path to the serial port", default=kDefaultSerialPort
|
||||||
parser.add_argument("--escape", action='store_true')
|
)
|
||||||
|
parser.add_argument("--baudrate", default=kDefaultBaudrate)
|
||||||
parser.add_argument("ws", help="URL of the websocket")
|
parser.add_argument("ws", help="URL of the websocket")
|
||||||
return parser.parse_args()
|
return parser.parse_args()
|
||||||
|
|
||||||
@@ -28,28 +45,23 @@ def slowwrite(device, data):
|
|||||||
|
|
||||||
def ws_thread(device, ws):
|
def ws_thread(device, ws):
|
||||||
while True:
|
while True:
|
||||||
data = ws.recv();
|
data = ws.recv()
|
||||||
slowwrite(device, data.replace(b'\n', b'\r\n'))
|
slowwrite(device, data.replace(b"\n", b"\r\n"))
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
args = parse_args()
|
args = parse_args()
|
||||||
device = serial.Serial(args.port, baudrate=args.baudrate)
|
device = getserial(**vars(args))
|
||||||
ws = websocket.create_connection(args.ws)
|
ws = websocket.create_connection(args.ws)
|
||||||
|
|
||||||
wst = threading.Thread(target=ws_thread, args=(device, ws), daemon=True)
|
wst = threading.Thread(target=ws_thread, args=(device, ws), daemon=True)
|
||||||
wst.start()
|
wst.start()
|
||||||
|
|
||||||
if args.escape:
|
|
||||||
device.write([0x03])
|
|
||||||
try:
|
|
||||||
device_thread(device, ws)
|
device_thread(device, ws)
|
||||||
except KeyboardInterrupt:
|
|
||||||
pass
|
|
||||||
finally:
|
|
||||||
if args.escape:
|
|
||||||
device.write([0x02])
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
try:
|
||||||
main()
|
main()
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
pass
|
||||||
|
Reference in New Issue
Block a user