diff --git a/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/base_hokuyo.launch b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/base_hokuyo.launch new file mode 100644 index 0000000000..e1c00a7a90 --- /dev/null +++ b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/base_hokuyo.launch @@ -0,0 +1,12 @@ + + + + + + + + + + + + diff --git a/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/check_hokuyo_scan_node.py b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/check_hokuyo_scan_node.py new file mode 100755 index 0000000000..e401697d6d --- /dev/null +++ b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/check_hokuyo_scan_node.py @@ -0,0 +1,113 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +import sys,subprocess,traceback +import rospy +import time +import os +from sensor_msgs.msg import Image, CompressedImage, LaserScan +from sound_play.msg import SoundRequest, SoundRequestAction, SoundRequestGoal +import actionlib +import roslib + +class HokuyoScanChecker: + def __init__(self): + rospy.init_node('hokuyo_scan_checker_node', anonymous=True) + self.no_topic_flag = False ## True if topic is currently not coming in. + self.once_topic_flag = False ## True if the topic has been received at least once since the start of node execution. + + self.launch_process = None + self.hokuyo_name = rospy.get_param('~hokuyo_name', 'base') + + # Create an Action client for the sound_play node + self.sound_client = actionlib.SimpleActionClient('/robotsound', SoundRequestAction) + self.sound_client.wait_for_server() + + # Threshold (in seconds) when a topic is not updated for a certain period of time + self.timeout_threshold = 10.0 + + # Time the topic was last updated + self.last_image_time = time.time() + + self.say_something("{} scan check start".format(self.hokuyo_name)) + + # Subscribing topic + self.topic_sub = rospy.Subscriber('/{}_scan'.format(self.hokuyo_name), LaserScan, self.topic_callback) + + def topic_callback(self, msg): + # Callback to be called when topic is updated + self.last_image_time = time.time() + if self.no_topic_flag or self.once_topic_flag==False: + self.no_topic_flag = False + self.once_topic_flag = True + self.say_something("{} scan topic is arrive.".format(self.hokuyo_name)) + + def check_timeout(self): + # Check if the topic has not been updated for a certain period of time + if time.time() - self.last_image_time > self.timeout_threshold: + if self.no_topic_flag: + return + else: + self.no_topic_flag = True + self.say_something("I haven't seen the {} scan topic for {} seconds.".format(self.hokuyo_name, self.timeout_threshold)) + self.restart_hokuyo() + + def say_something(self, text): + # Let the robot talk + rospy.loginfo(text) + + # Create a SoundRequestGoal message + sound_goal = SoundRequestGoal() + sound_goal.sound_request.sound = SoundRequest.SAY + sound_goal.sound_request.command = SoundRequest.PLAY_ONCE + sound_goal.sound_request.volume = 1.0 + sound_goal.sound_request.arg = text + + # Send the SoundRequestGoal to the sound_play node + self.sound_client.send_goal(sound_goal) + + # Wait for the result (you can add timeout if needed) + self.sound_client.wait_for_result() + + def restart_hokuyo(self): + rospy.logerr("Restarting {} hokuyo".format(self.hokuyo_name)) + retcode = -1 + if self.launch_process: ## Force termination if launch process exists + self.launch_process.terminate() + self.launch_process.wait() + try: + # 1. kill hokuyo node + retcode = subprocess.call('rosnode kill /{}_hokuyo_node'.format(self.hokuyo_name), shell=True) + retcode = subprocess.call('pkill -f {}_hokuyo_node'.format(self.hokuyo_name), shell=True) + rospy.loginfo("Killed {} hokuyo node".format(self.hokuyo_name)) + time.sleep(10) + + # 2. reset hokuyo + package_path = roslib.packages.get_pkg_dir('jsk_pr2_startup') + script_path = os.path.join(package_path, 'jsk_pr2_sensors/hokuyo_reset_scripts') + retcode = subprocess.call('{}/upgrade /etc/ros/sensors/{}_hokuyo {}/reset.cmd'.format(script_path, self.hokuyo_name, script_path), shell=True) + self.say_something("Reset {} hokuyo".format(self.hokuyo_name)) + time.sleep(10) + + # 3 Restarting hokuyo node + os.environ['ROS_ENV_LOADER'] = '/home/applications/ros/noetic/devel/env.sh' + self.launch_process = subprocess.Popen(['roslaunch', 'jsk_pr2_startup', '{}_hokuyo.launch'.format(self.hokuyo_name)], env=os.environ) + rospy.loginfo("Restart {} hokuyo node".format(self.hokuyo_name)) + time.sleep(30) + rospy.loginfo("Restarting {} hokuyo node is Done".format(self.hokuyo_name)) + + except Exception as e: + rospy.logerr('[%s] Unable to kill %s hokuyo node, caught exception:\n%s', self.__class__.__name__, self.hokuyo_name, traceback.format_exc()) + + def run(self): + rate = rospy.Rate(1) # Router plate: 1 Hz + while not rospy.is_shutdown(): + self.check_timeout() + rate.sleep() + +if __name__ == '__main__': + try: + topic_checker = HokuyoScanChecker() + topic_checker.run() + except rospy.ROSInterruptException: + pass diff --git a/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/hokuyo_reset_scripts/reset.cmd b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/hokuyo_reset_scripts/reset.cmd new file mode 100644 index 0000000000..b84f61c508 --- /dev/null +++ b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/hokuyo_reset_scripts/reset.cmd @@ -0,0 +1,5 @@ +*** 1 O 4 +$WP +*** 2 O 5 +$WCE +*** 3 E 0 diff --git a/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/hokuyo_reset_scripts/upgrade b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/hokuyo_reset_scripts/upgrade new file mode 100755 index 0000000000..edaff4dbdf Binary files /dev/null and b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/hokuyo_reset_scripts/upgrade differ diff --git a/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/tilt_hokuyo.launch b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/tilt_hokuyo.launch new file mode 100644 index 0000000000..49fc13fe05 --- /dev/null +++ b/jsk_pr2_robot/jsk_pr2_startup/jsk_pr2_sensors/tilt_hokuyo.launch @@ -0,0 +1,12 @@ + + + + + + + + + + + + diff --git a/jsk_pr2_robot/jsk_pr2_startup/pr2_bringup.launch b/jsk_pr2_robot/jsk_pr2_startup/pr2_bringup.launch index 823f824d0e..187e60041b 100644 --- a/jsk_pr2_robot/jsk_pr2_startup/pr2_bringup.launch +++ b/jsk_pr2_robot/jsk_pr2_startup/pr2_bringup.launch @@ -77,24 +77,22 @@ - - - - - - - - + + + + + + - - - - - - - - + + + + + +