#!/usr/bin/env python3
"""
RobotX Communications: A node that handles message communications
with the Technical Director server for the RobotX Communication Protocol
"""
import datetime
import socket
import threading
from enum import IntEnum
import rospy
from geometry_msgs.msg import PointStamped
from mil_tools import thread_lock
from nav_msgs.msg import Odometry
from navigator_msgs.msg import ScanTheCode
from navigator_msgs.srv import (
MessageDetectDock,
MessageDetectDockRequest,
MessageDetectDockResponse,
MessageEntranceExitGate,
MessageEntranceExitGateRequest,
MessageEntranceExitGateResponse,
MessageFindFling,
MessageFindFlingRequest,
MessageFindFlingResponse,
MessageFollowPath,
MessageFollowPathRequest,
MessageFollowPathResponse,
MessageUAVReplenishment,
MessageUAVReplenishmentRequest,
MessageUAVReplenishmentResponse,
MessageUAVSearchReport,
MessageUAVSearchReportRequest,
MessageUAVSearchReportResponse,
MessageWildlifeEncounter,
MessageWildlifeEncounterRequest,
MessageWildlifeEncounterResponse,
)
from navigator_robotx_comms.navigator_robotx_comms import (
RobotXDetectDockMessage,
RobotXEntranceExitGateMessage,
RobotXFindFlingMessage,
RobotXFollowPathMessage,
RobotXHeartbeatMessage,
RobotXScanCodeMessage,
RobotXUAVReplenishmentMessage,
RobotXUAVSearchReportMessage,
RobotXWildlifeEncounterMessage,
)
from ros_alarms import AlarmListener
from ros_alarms_msgs.msg import Alarm
from std_msgs.msg import String
lock = threading.Lock()
class SystemModes(IntEnum):
"""
Enumerates constants of friendly system mode names to ints
"""
KILLED = 3
AUTONOMOUS = 2
REMOTE_CONTROLLED = 1
[docs]class RobotXStartServices:
"""
Initializes services and subscribes to necessary publishers in order to facilitate
the transmission of messages between the robot client and the Technical Director
server.
This class is part of a node that should be launched when communication is necessary
with official AUVSI software.
Attributes:
robotx_client (RobotXClient): The client used to connect with the Technical
Director server.
system_mode (int): The mode of the system.
wrench (str): The current wrench mode of the system. For example, ``autonomous``.
"""
def __init__(self):
# define all variables for subscribers
self.gps_array = None
self.odom = None
self.uav_status = 1
self.system_mode = None
self.wrench = None
self.kill = False
# define delimiter for messages
self.delim = ","
# define parameters
self.team_id = rospy.get_param("~team_id")
self.td_ip = rospy.get_param("~td_ip")
self.td_port = rospy.get_param("~td_port")
self.use_test_data = rospy.get_param("~use_test_data")
# time last called
self.time_last_entrance_exit = None
self.time_last_follow_path = None
self.time_last_wildlife_encounter = None
self.time_last_find_fling = None
self.time_last_uav_replenishment = None
self.time_last_uav_search_report = None
self.time_last_scan_code = None
self.time_last_detect_dock = None
# initialize connection to server
self.robotx_client = RobotXClient(self.td_ip, self.td_port)
self.robotx_client.connect()
# setup all message types
self.robotx_heartbeat_message = RobotXHeartbeatMessage()
self.robotx_entrance_exit_gate_message = RobotXEntranceExitGateMessage()
self.robotx_scan_code_message = RobotXScanCodeMessage()
self.robotx_detect_dock_message = RobotXDetectDockMessage()
self.robotx_follow_path_message = RobotXFollowPathMessage()
self.robotx_wildlife_encounter_message = RobotXWildlifeEncounterMessage()
self.robotx_find_fling_message = RobotXFindFlingMessage()
self.robotx_uav_replenishment_message = RobotXUAVReplenishmentMessage()
self.robotx_uav_search_report_message = RobotXUAVSearchReportMessage()
# setup all subscribers
rospy.Subscriber("lla", PointStamped, self.gps_coord_callback)
rospy.Subscriber("odom", Odometry, self.gps_odom_callback)
rospy.Subscriber("/wrench/selected", String, self.wrench_callback)
rospy.Subscriber("/scan_the_code", ScanTheCode, self.scan_the_code_callback)
# TODO: setup subscriber for uav status callback update
# track kill state for inferring system mode
self.kill_listener = AlarmListener("kill", self.kill_callback)
self.kill_listener.wait_for_server()
# setup all services
self.service_entrance_exit_gate_message = rospy.Service(
"entrance_exit_gate_message",
MessageEntranceExitGate,
self.handle_entrance_exit_gate_message,
)
self.service_follow_path_message = rospy.Service(
"follow_path_message",
MessageFollowPath,
self.handle_follow_path_message,
)
self.service_wildlife_encounter_message = rospy.Service(
"wildlife_encounter_message",
MessageWildlifeEncounter,
self.handle_wildlife_encounter_message,
)
self.service_detect_dock_message = rospy.Service(
"detect_dock_message",
MessageDetectDock,
self.handle_detect_dock_message,
)
self.service_find_fling_message = rospy.Service(
"find_fling_message",
MessageFindFling,
self.handle_find_fling_message,
)
self.service_uav_replenishment_message = rospy.Service(
"uav_replenishment_message",
MessageUAVReplenishment,
self.handle_uav_replenishment_message,
)
self.service_uav_search_report_message = rospy.Service(
"uav_search_report_message",
MessageUAVSearchReport,
self.handle_uav_search_report_message,
)
# start sending heartbeat every second
rospy.Timer(rospy.Duration(1), self.handle_heartbeat_message)
[docs] def update_system_mode(self) -> None:
"""
Sets :attr:`.system_mode` according to whether the kill is ``True`` and the mode
of the boat.
"""
if self.kill is True:
self.system_mode = SystemModes.KILLED
elif self.wrench == "autonomous" or self.wrench == "/wrench/autonomous":
self.system_mode = SystemModes.AUTONOMOUS
else:
self.system_mode = SystemModes.REMOTE_CONTROLLED
[docs] def wrench_callback(self, wrench: String) -> None:
"""
Updates the :attr:`.wrench` attribute with the most recent wrench value.
"""
self.wrench = wrench.data
[docs] def kill_callback(self, alarm: Alarm):
"""
Updates the :attr:`.kill` attribute depending on whether the most recent
alarm received by the boat was raised or not.
"""
self.kill = alarm.raised
[docs] def gps_coord_callback(self, lla: PointStamped) -> None:
"""
Updates the :attr:`.gps_array` attribute with the most recent :class:`PointStamped`
message received.
"""
self.gps_array = lla
[docs] def gps_odom_callback(self, odom: Odometry):
"""
Stores the most recent :class:`Odometry` message.
"""
self.odom = odom
[docs] def uav_status_callback(self, uav_status: int):
"""
Stores the most recent AUV status experienced by the boat.
"""
self.uav_status = uav_status
[docs] def system_mode_callback(self, system_mode: int):
"""
Sets the class' :attr:`system_mode` attribute when given the most recent
system mode.
"""
self.system_mode = system_mode
[docs] def scan_the_code_callback(self, scan_the_code: ScanTheCode):
"""
Handles the ScanTheCode message requests by calling :meth:`.handle_scan_code_message`
with the color pattern specified in the message.
Args:
scan_the_code (ScanTheCode): The message to format and send to the Technical
Director server.
"""
self.handle_scan_code_message(scan_the_code.color_pattern)
[docs] def handle_heartbeat_message(self, _) -> None:
"""
Constructs a heartbeat message according to a timer and sends the formatted
message to the AUVSI Technical Director station.
"""
self.update_system_mode()
edt_date_time = self.get_edt_date_time()
message = self.robotx_heartbeat_message.to_string(
self.delim,
self.team_id,
edt_date_time,
self.gps_array,
self.odom,
self.uav_status,
self.system_mode,
self.use_test_data,
)
self.robotx_client.send_message(message)
[docs] def handle_entrance_exit_gate_message(
self,
data: MessageEntranceExitGateRequest,
) -> MessageEntranceExitGateResponse:
"""
Handles requests to make messages to use in the Entrance and Exit Gate
mission.
Args:
data (MessageEntranceExitGateRequest): The request to the service.
Returns:
MessageEntranceExitGateResponse: The response from the service. The response
contains the message needed to send to AUVSI.
"""
if self.time_last_entrance_exit is not None:
seconds_elapsed = rospy.get_time() - self.time_last_entrance_exit
if seconds_elapsed < 1:
rospy.sleep(1 - seconds_elapsed)
self.time_last_entrance_exit = rospy.get_time()
edt_date_time = self.get_edt_date_time()
message = self.robotx_entrance_exit_gate_message.to_string(
self.delim,
self.team_id,
edt_date_time,
data,
self.use_test_data,
)
self.robotx_client.send_message(message)
return MessageEntranceExitGateResponse(message)
[docs] def handle_follow_path_message(
self,
data: MessageFollowPathRequest,
) -> MessageFollowPathResponse:
"""
Handles requests to make messages to use in the Follow Path Mission
Args:
data (MessageFollowPathRequest): The request to the service.
Returns:
MessageFollowPathResponse: The response from the service. The response
contains the message needed to send to AUVSI.
"""
if self.time_last_follow_path is not None:
seconds_elapsed = rospy.get_time() - self.time_last_follow_path
if seconds_elapsed < 1:
rospy.sleep(1 - seconds_elapsed)
self.time_last_follow_path = rospy.get_time()
edt_date_time = self.get_edt_date_time()
message = self.robotx_follow_path_message.to_string(
self.delim,
self.team_id,
edt_date_time,
data,
self.use_test_data,
)
self.robotx_client.send_message(message)
return MessageFollowPathResponse(message)
[docs] def handle_wildlife_encounter_message(
self,
data: MessageWildlifeEncounterRequest,
) -> MessageWildlifeEncounterResponse:
"""
Handles requests to make messages to use in the Wildlife Encounter Mission
Args:
data (MessageWildlifeEncounterRequest): The request to the service.
Returns:
MessageWildlifeEncounterResponse: The response from the service. The response
contains the message needed to send to AUVSI.
"""
if self.time_last_wildlife_encounter is not None:
seconds_elapsed = rospy.get_time() - self.time_last_wildlife_encounter
if seconds_elapsed < 1:
rospy.sleep(1 - seconds_elapsed)
self.time_last_wildlife_encounter = rospy.get_time()
edt_date_time = self.get_edt_date_time()
message = self.robotx_wildlife_encounter_message.to_string(
self.delim,
self.team_id,
edt_date_time,
data,
self.use_test_data,
)
self.robotx_client.send_message(message)
return MessageWildlifeEncounterResponse(message)
[docs] def handle_scan_code_message(self, color_pattern: str) -> None:
"""
Handles the color pattern reported over the topic dedicated to the Scan the Code
mission.
The color is encoded into a message and sent to the AUVSI Technical Director
server.
Args:
color_pattern (str): The color pattern reported by the handled message
in the Scan the Code topic.
"""
if self.time_last_scan_code is not None:
seconds_elapsed = rospy.get_time() - self.time_last_scan_code
if seconds_elapsed < 1:
rospy.sleep(1 - seconds_elapsed)
self.time_last_scan_code = rospy.get_time()
edt_date_time = self.get_edt_date_time()
message = self.robotx_scan_code_message.to_string(
self.delim,
self.team_id,
edt_date_time,
color_pattern,
self.use_test_data,
)
self.robotx_client.send_message(message)
[docs] def handle_detect_dock_message(
self,
data: MessageDetectDockRequest,
) -> MessageDetectDockResponse:
"""
Handles requests to make messages to use in the Detect Dock Mission
Args:
data (MessageDetectDockRequest): The request to the service.
Returns:
MessageDetectDockResponse: The response from the service. The response
contains the message needed to send to AUVSI.
"""
if self.time_last_detect_dock is not None:
seconds_elapsed = rospy.get_time() - self.time_last_detect_dock
if seconds_elapsed < 1:
rospy.sleep(1 - seconds_elapsed)
self.time_last_detect_dock = rospy.get_time()
edt_date_time = self.get_edt_date_time()
message = self.robotx_detect_dock_message.to_string(
self.delim,
self.team_id,
edt_date_time,
data,
self.use_test_data,
)
self.robotx_client.send_message(message)
return MessageDetectDockResponse(message)
[docs] def handle_find_fling_message(
self,
data: MessageFindFlingRequest,
) -> MessageFindFlingResponse:
"""
Handles requests to make messages to use in the Find Fling Mission
Args:
data (MessageFindFlingRequest): The request to the service.
Returns:
MessageFindFlingResponse: The response from the service. The response
contains the message needed to send to AUVSI.
"""
if self.time_last_find_fling is not None:
seconds_elapsed = rospy.get_time() - self.time_last_find_fling
if seconds_elapsed < 1:
rospy.sleep(1 - seconds_elapsed)
self.time_last_find_fling = rospy.get_time()
edt_date_time = self.get_edt_date_time()
message = self.robotx_find_fling_message.to_string(
self.delim,
self.team_id,
edt_date_time,
data,
self.use_test_data,
)
self.robotx_client.send_message(message)
return MessageFindFlingResponse(message)
[docs] def handle_uav_replenishment_message(
self,
data: MessageUAVReplenishmentRequest,
) -> MessageUAVReplenishmentResponse:
"""
Handles requests to make messages to use in the UAV Replenishment Mission
Args:
data (MessageUAVReplenishmentRequest): The request to the service.
Returns:
MessageUAVReplenishmentResponse: The response from the service. The response
contains the message needed to send to AUVSI.
"""
if self.time_last_uav_replenishment is not None:
seconds_elapsed = rospy.get_time() - self.time_last_uav_replenishment
if seconds_elapsed < 1:
rospy.sleep(1 - seconds_elapsed)
self.time_last_uav_replenishment = rospy.get_time()
edt_date_time = self.get_edt_date_time()
message = self.robotx_uav_replenishment_message.to_string(
self.delim,
self.team_id,
edt_date_time,
data,
self.use_test_data,
)
self.robotx_client.send_message(message)
return MessageUAVReplenishmentResponse(message)
[docs] def handle_uav_search_report_message(
self,
data: MessageUAVSearchReportRequest,
) -> MessageUAVSearchReportResponse:
"""
Handles requests to make messages to use in the UAV Search Report Mission
Args:
data (MessageUAVSearchReportRequest): The request to the service.
Returns:
MessageUAVSearchReportResponse: The response from the service. The response
contains the message needed to send to AUVSI.
"""
if self.time_last_uav_search_report is not None:
seconds_elapsed = rospy.get_time() - self.time_last_uav_search_report
if seconds_elapsed < 1:
rospy.sleep(1 - seconds_elapsed)
self.time_last_uav_search_report = rospy.get_time()
edt_date_time = self.get_edt_date_time()
message = self.robotx_uav_search_report_message.to_string(
self.delim,
self.team_id,
edt_date_time,
data,
self.use_test_data,
)
self.robotx_client.send_message(message)
return MessageUAVSearchReportResponse(message)
[docs] def get_edt_date_time(self) -> str:
"""
Gets the current time in AEDT in the format of ``%d%m%y{self.delim}%H%M%S``.
This is the format specified by AUVSI to use in messages.
Returns:
str: The constructed string representing the date and time.
"""
# EDT is 11 hours ahead of UTC
edt_time = datetime.datetime.utcnow() - datetime.timedelta(hours=5)
date_string = edt_time.strftime("%d%m%y")
time_string = edt_time.strftime("%H%M%S")
return date_string + self.delim + time_string
[docs]class RobotXClient:
"""
Handles communication with Technical Director server at the AUVSI RobotX competition.
This class is generally handled by :class:`RobotXStartServices`.
Attributes:
tcp_port (int): The TCP port allotted for communication between MIL and the
Technical Director server at the competition site.
connected (bool): Whether a connection is intact between the client and the
Technical Director server.
socket_connection (Optional[socket.socket]): The socket connection. This is
initially set to ``None``, until it is made in :meth:`.connect`.
"""
def __init__(self, tcp_ip: str, tcp_port: int):
"""
Args:
tcp_ip (str): The IP address used by the Technical Director server to connect to.
tcp_port (int): The port used by the Technical Director server that MIL
is asked to connect to.
"""
while True:
try:
self.tcp_ip = socket.gethostbyname(tcp_ip)
break
except socket.gaierror as e:
rospy.logwarn(f"Failed to resolved {tcp_ip}: {e}")
rospy.sleep(1.0)
continue
self.tcp_port = tcp_port
self.connected = False
self.socket_connection = None
[docs] def connect(self) -> None:
"""
Attempts to establish a socket connection at the given IP and port number
if no connection has already been established.
If a connection can not be established, then the method sleeps for 2 seconds
before retrying to establish a connection.
"""
if not self.connected:
rospy.loginfo(
f"Attempting Connection to TD Server at {self.tcp_ip}:{self.tcp_port}",
)
while not self.connected and not rospy.is_shutdown():
# recreate socket
self.socket_connection = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# attempt to reconnect, otherwise sleep for 2 seconds
try:
self.socket_connection.connect((self.tcp_ip, self.tcp_port))
self.connected = True
rospy.loginfo("Connection to TD Server Successful")
except OSError:
rospy.sleep(2)
@thread_lock(lock)
def send_message(self, message: str) -> None:
"""
Sends a message over the established socket connection. This method is thread-locked.
If any error occurs in the transmission of the packet, then :attr:`.connection`
is immediately set to ``False``, and a connection is attempted to be re-established
through :meth:`.connect`.
Args:
message (bytes): The message to send over the connection.
"""
while not rospy.is_shutdown():
try:
self.socket_connection.send(bytes(message, "ascii"))
break
except OSError:
rospy.loginfo("Connection to TD Server Lost")
self.connected = False
self.connect()
if __name__ == "__main__":
rospy.init_node("robotx_comms_client")
robotx_start_services = RobotXStartServices()
rospy.spin()