From 2f3e091b7b575178dac730d6fcb3b66b51afc635 Mon Sep 17 00:00:00 2001
From: MarcinPlaza <109978177+MarcinPlaza@users.noreply.github.com>
Date: Sun, 17 Nov 2024 20:23:05 -0500
Subject: [PATCH] first attempt at async driver (with errors!)
---
.../electrical_protocol/CMakeLists.txt | 2 +
.../electrical_protocol/Async_readme.txt | 26 +++++
.../electrical_protocol/__init__.py | 1 +
.../electrical_protocol/async_driver.py | 68 +++++++-----
.../electrical_protocol/driver.py | 2 +
.../drivers/electrical_protocol/package.xml | 1 +
.../test/async_calculator_device.py | 104 ++++++++++++++++++
.../test/async_simulated.test | 7 ++
.../test/async_test_simulated_basic.py | 99 +++++++++++++++++
.../test/calculator_device.py | 10 +-
.../test/test_simulated_basic.py | 6 +-
11 files changed, 293 insertions(+), 33 deletions(-)
create mode 100644 mil_common/drivers/electrical_protocol/electrical_protocol/Async_readme.txt
create mode 100755 mil_common/drivers/electrical_protocol/test/async_calculator_device.py
create mode 100644 mil_common/drivers/electrical_protocol/test/async_simulated.test
create mode 100755 mil_common/drivers/electrical_protocol/test/async_test_simulated_basic.py
diff --git a/mil_common/drivers/electrical_protocol/CMakeLists.txt b/mil_common/drivers/electrical_protocol/CMakeLists.txt
index 3424e2cb3..9a00e3384 100644
--- a/mil_common/drivers/electrical_protocol/CMakeLists.txt
+++ b/mil_common/drivers/electrical_protocol/CMakeLists.txt
@@ -3,7 +3,9 @@ project(electrical_protocol)
find_package(catkin REQUIRED COMPONENTS
rospy
serial
+ axros
)
+
catkin_python_setup()
catkin_package()
diff --git a/mil_common/drivers/electrical_protocol/electrical_protocol/Async_readme.txt b/mil_common/drivers/electrical_protocol/electrical_protocol/Async_readme.txt
new file mode 100644
index 000000000..1180c480f
--- /dev/null
+++ b/mil_common/drivers/electrical_protocol/electrical_protocol/Async_readme.txt
@@ -0,0 +1,26 @@
+Everything currently labelled async is unifinished
+and not fully working.
+
+The last issue I had was the simulated device not
+being able to connect to the test, all async files are
+commented out.
+
+You can run the async test with the following command:
+"rostest --text electrical_protocol async_simulated.test"
+
+If you have any questions or confusion about the mess
+I've made here feel free to reach out: marcinplaza@ufl.edu
+
+ps: when starting you will most likely have to make
+async_test_simulated_basic.py / async_driver.py executable
+example: chmod +x async_test_simulated_basic.py (makes it executable)
+
+some helpful sites:
+axros examples
+https://mil.ufl.edu/docs/reference/axros/examples.html
+
+axros reference
+https://mil.ufl.edu/docs/reference/axros/api.html#service
+
+asyncio reference
+https://docs.python.org/3/library/asyncio.html
diff --git a/mil_common/drivers/electrical_protocol/electrical_protocol/__init__.py b/mil_common/drivers/electrical_protocol/electrical_protocol/__init__.py
index 0bec1b8c3..6a0e54793 100644
--- a/mil_common/drivers/electrical_protocol/electrical_protocol/__init__.py
+++ b/mil_common/drivers/electrical_protocol/electrical_protocol/__init__.py
@@ -11,5 +11,6 @@
class can be used, which supports packets built through this library.
"""
+from .async_driver import AsyncSerialDevice
from .driver import ROSSerialDevice
from .packet import AckPacket, ChecksumException, NackPacket, Packet
diff --git a/mil_common/drivers/electrical_protocol/electrical_protocol/async_driver.py b/mil_common/drivers/electrical_protocol/electrical_protocol/async_driver.py
index 107ea362f..b9e49eeb0 100644
--- a/mil_common/drivers/electrical_protocol/electrical_protocol/async_driver.py
+++ b/mil_common/drivers/electrical_protocol/electrical_protocol/async_driver.py
@@ -1,3 +1,4 @@
+'''
from __future__ import annotations
import abc
@@ -6,7 +7,8 @@
from typing import Any, Generic, TypeVar, Union, cast, get_args, get_origin
import axros
-import rospy
+
+# import rospy
import serial
import serial_asyncio
@@ -23,13 +25,17 @@ def __init__(self, device: AsyncSerialDevice):
self.byte_count = 0
self.buffer = b""
self.expected_count = None
+ print("in init?")
def connection_made(self, transport: asyncio.BaseTransport) -> None:
self.transport = transport
+
+ print("port opened", transport)
self.transport.serial.reset_input_buffer() # type: ignore
self.transport.serial.reset_output_buffer() # type: ignore
def data_received(self, data: bytes) -> None:
+ print("data received?")
# Either wait to hear the next start byte, or keep accumulating bytes
# for the next packet
if self.expected_count is None:
@@ -114,7 +120,7 @@ def __init_subclass__(cls) -> None:
cls._send_T = get_args(cls.__orig_bases__[0])[0] # type: ignore
cls._recv_T = get_args(cls.__orig_bases__[0])[1] # type: ignore
- async def connect(self, port: str, baudrate: int) -> None:
+ async def connect(self, port: str, baudrate: int, loop) -> None:
"""
Connects to the port with the given baudrate. If the device is already
connected, the input and output buffers will be flushed.
@@ -123,34 +129,40 @@ async def connect(self, port: str, baudrate: int) -> None:
port (str): The serial port to connect to.
baudrate (int): The baudrate to connect with.
"""
+ print("3")
self.port = port
self.baudrate = baudrate
- if self.transport:
- raise RuntimeError("Device is already connected.")
+
+ # if self.transport:
+ # raise RuntimeError("Device is already connected.")
+ print("the serial connection runs after this?")
self.transport, self.protocol = await serial_asyncio.create_serial_connection(
- asyncio.get_event_loop(),
+ loop,
DeviceProtocol,
port,
baudrate=baudrate,
)
- def close(self) -> None:
+ print("the serial connection creator finished?")
+
+ async def close(self) -> None:
"""
Closes the serial device.
"""
- rospy.loginfo("Closing serial device...")
+ print("Closing serial device...")
if not self.transport:
raise RuntimeError("Device is not connected.")
else:
with contextlib.suppress(OSError):
if self.transport.serial.in_waiting: # type: ignore
- rospy.logwarn(
+ print(
"Shutting down device, but packets were left in buffer...",
)
self.transport.close()
+
self.transport = None
- def write(self, data: bytes) -> None:
+ async def write(self, data: bytes) -> None:
"""
Writes a series of raw bytes to the device. This method should rarely be
used; using :meth:`~.send_packet` is preferred because of the guarantees
@@ -164,21 +176,22 @@ def write(self, data: bytes) -> None:
self.transport.write(data)
def send_packet(self, packet: SendPackets) -> None:
+ print("4")
"""
Sends a given packet to the device.
Arguments:
packet (:class:`~.Packet`): The packet to send.
"""
- self.write(bytes(packet))
+ asyncio.create_task(self.write(bytes(packet)))
- def _read_from_stream(self) -> bytes:
+ async def _read_from_stream(self) -> bytes:
# Read until SOF is encourntered in case buffer contains the end of a previous packet
- if not self.device:
+ if not self.transport:
raise RuntimeError("Device is not connected.")
sof = None
for _ in range(10):
- sof = self.device.read(1)
+ sof = self.transport.serial.read(1)
if not len(sof):
continue
sof_int = int.from_bytes(sof, byteorder="big")
@@ -188,13 +201,13 @@ def _read_from_stream(self) -> bytes:
raise TimeoutError("No SOF received in one second.")
sof_int = int.from_bytes(sof, byteorder="big")
if sof_int != SYNC_CHAR_1:
- rospy.logerr("Where da start char at?")
+ print("Where da start char at?")
data = sof
# Read sync char 2, msg ID, subclass ID
- data += self.device.read(3)
- length = self.device.read(2) # read payload length
+ data += self.transport.serial.read(3)
+ length = self.transport.serial.read(2) # read payload length
data += length
- data += self.device.read(
+ data += self.transport.serial.read(
int.from_bytes(length, byteorder="little") + 2,
) # read data and checksum
return data
@@ -208,14 +221,14 @@ def _correct_type(self, provided: Any) -> bool:
else:
return isinstance(provided, self._recv_T)
- def _read_packet(self) -> bool:
+ async def _read_packet(self) -> bool:
if not self.device:
raise RuntimeError("Device is not connected.")
try:
if not self.is_open() or self.device.in_waiting == 0:
return False
if self.device.in_waiting > 200:
- rospy.logwarn_throttle(
+ print(
0.5,
"Packets are coming in much quicker than expected, upping rate...",
)
@@ -224,33 +237,33 @@ def _read_packet(self) -> bool:
assert isinstance(packed_packet, bytes)
packet = Packet.from_bytes(packed_packet)
except serial.SerialException as e:
- rospy.logerr(f"Error reading packet: {e}")
+ print(f"Error reading packet: {e}")
return False
except OSError:
- rospy.logerr_throttle(1, "Cannot read from serial device.")
+ print(1, "Cannot read from serial device.")
return False
if not self._correct_type(packet):
- rospy.logerr(
+ print(
f"Received unexpected packet: {packet} (expected: {self._recv_T})",
)
return False
packet = cast(RecvPackets, packet)
- self.on_packet_received(packet)
+ await self.on_packet_received(packet)
return True
- def _process_buffer(self, _: rospy.timer.TimerEvent) -> None:
+ async def _process_buffer(self, _: rospy.timer.TimerEvent) -> None:
if not self.is_open():
return
try:
- self._read_packet()
+ await self._read_packet()
except Exception as e:
- rospy.logerr(f"Error reading recent packet: {e}")
+ print(f"Error reading recent packet: {e}")
import traceback
traceback.print_exc()
@abc.abstractmethod
- def on_packet_received(self, packet: RecvPackets) -> None:
+ async def on_packet_received(self, packet: RecvPackets) -> None:
"""
Abstract method to be implemented by subclasses for handling packets
sent by the physical electrical board.
@@ -259,3 +272,4 @@ def on_packet_received(self, packet: RecvPackets) -> None:
packet (:class:`~.Packet`): The packet that is received.
"""
pass
+'''
diff --git a/mil_common/drivers/electrical_protocol/electrical_protocol/driver.py b/mil_common/drivers/electrical_protocol/electrical_protocol/driver.py
index 5e7c669ae..01cc371ba 100644
--- a/mil_common/drivers/electrical_protocol/electrical_protocol/driver.py
+++ b/mil_common/drivers/electrical_protocol/electrical_protocol/driver.py
@@ -75,6 +75,7 @@ def connect(self, port: str, baudrate: int) -> None:
port (str): The serial port to connect to.
baudrate (int): The baudrate to connect with.
"""
+ rospy.loginfo("3")
self.port = port
self.baudrate = baudrate
self.device = serial.Serial(port, baudrate, timeout=0.1)
@@ -113,6 +114,7 @@ def write(self, data: bytes) -> None:
self.device.write(data)
def send_packet(self, packet: SendPackets) -> None:
+ rospy.loginfo("4")
"""
Sends a given packet to the device.
diff --git a/mil_common/drivers/electrical_protocol/package.xml b/mil_common/drivers/electrical_protocol/package.xml
index 07cf05e65..86c327c5a 100644
--- a/mil_common/drivers/electrical_protocol/package.xml
+++ b/mil_common/drivers/electrical_protocol/package.xml
@@ -6,6 +6,7 @@
cameron.brown
MIT
catkin
+ axros
rospy
serial
diff --git a/mil_common/drivers/electrical_protocol/test/async_calculator_device.py b/mil_common/drivers/electrical_protocol/test/async_calculator_device.py
new file mode 100755
index 000000000..9515f6f64
--- /dev/null
+++ b/mil_common/drivers/electrical_protocol/test/async_calculator_device.py
@@ -0,0 +1,104 @@
+#!/usr/bin/env python3
+"""
+from __future__ import annotations
+
+import asyncio
+from dataclasses import dataclass
+
+# from threading import Event, Thread
+from typing import Union
+
+import uvloop
+from axros import NodeHandle
+from electrical_protocol import AsyncSerialDevice, Packet
+from std_msgs.msg import Float32, String
+from std_srvs.srv import Empty, EmptyRequest, EmptyResponse
+
+
+@dataclass
+class RequestAddPacket(Packet, class_id=0x37, subclass_id=0x00, payload_format=" EmptyResponse:
+ if self.transport == None:
+ print("transport is not with us")
+ print("4")
+ self.num_one, self.num_two = self.i, 1000 - self.i
+ self.i += 1
+ self.send_packet(
+ RequestAddPacket(number_one=self.num_one, number_two=self.num_two),
+ )
+ return EmptyResponse()
+
+ def on_packet_received(self, packet) -> None:
+ self.answer_topic.publish(Float32(data=packet.result))
+ self.next_packet.set()
+
+
+async def main():
+ nh, remaining_args = NodeHandle.from_argv_with_remaining(
+ "async_calculator_device", anonymous=True,
+ )
+ async with nh:
+ print("Starting AsyncCalculatorDevice...")
+ device = AsyncCalculatorDevice(nh)
+
+ await device.setup()
+
+
+if __name__ == "__main__":
+ uvloop.install()
+ asyncio.run(main())
+"""
diff --git a/mil_common/drivers/electrical_protocol/test/async_simulated.test b/mil_common/drivers/electrical_protocol/test/async_simulated.test
new file mode 100644
index 000000000..87745fa54
--- /dev/null
+++ b/mil_common/drivers/electrical_protocol/test/async_simulated.test
@@ -0,0 +1,7 @@
+
+
diff --git a/mil_common/drivers/electrical_protocol/test/async_test_simulated_basic.py b/mil_common/drivers/electrical_protocol/test/async_test_simulated_basic.py
new file mode 100755
index 000000000..cbbe9d0d5
--- /dev/null
+++ b/mil_common/drivers/electrical_protocol/test/async_test_simulated_basic.py
@@ -0,0 +1,99 @@
+#!/usr/bin/env python3
+"""
+import os
+import pty
+import unittest
+from dataclasses import dataclass
+
+import rospy
+import rostest
+from electrical_protocol import Packet
+from std_msgs.msg import Float32, String
+from std_srvs.srv import Empty
+
+
+@dataclass
+class RequestAddPacket(Packet, class_id=0x37, subclass_id=0x00, payload_format="