diff --git a/doc/jsk_tools/scripts/audible_warning.md b/doc/jsk_tools/scripts/audible_warning.md new file mode 100644 index 000000000..95fb4d3a9 --- /dev/null +++ b/doc/jsk_tools/scripts/audible_warning.md @@ -0,0 +1,156 @@ +audible\_warning.py +======================= + +## What is this + +This is a general node that subscribes to `/diagnostics_agg` to speak the error content. +Robots using diagnostics can use this node. + +## Target Action + +* `/robotsound` (`sound_play/SoundRequestAction`) + + Target action name. + + If it is a different server name, please remap it like ``. + +## Subscribing Topics + +* `/diagnostics_agg` (`diagnostic_msgs/DiagnosticArray`) + + Aggregated diagnostics. + + +## Parameter + +* `~speak_rate` (`Float`, default: `1.0 / 100.0`) + + Rate of speak loop. If `~wait_speak` is `True`, wait until a robot finish speaking. + +* `~speak_interval` (`Float`, default: `120.0`) + + The same error will not be spoken until this number of seconds has passed. + +* `~volume` (`Float`, default: `1.0`) + + Volume of speaking. + +* `~language` (`String`, default: `""`) + + Language parameters for `arg2` in `sound_play/SoundRequestAction`. + +* `~wait_speak` (`Bool`, default: `True`) + + If `True`, wait until finish saying one error. + +* `~seconds_to_start_speaking` (`Float`, default: `0.0`) + + It is the time to wait for the node to speak after it has started. + + This is useful for ignoring errors that occur when the robot starts. + +* `~wait_speak_duration_time` (`Float`, default: `30.0`) + + Waiting time in `robotsound` action. + +* `~enable` (`Bool`, default: `True`) + + If `True`, speak diagnositcs. If `False`, this node don't speak. + +* `~speak_ok` (`Bool`, default: `False`) + + If `True`, speak ok level diagnostics. + +* `~speak_warn` (`Bool`, default: `True`) + + If `True`, speak warning level diagnostics. + +* `~speak_error` (`Bool`, default: `True`) + + If `True`, speak error level diagnostics. + +* `~speak_stale` (`Bool`, default: `True`) + + If `True`, speak stale level diagnostics. + +* `~speak_when_runstopped` (`Bool`, default: `True`) + + If `True`, speak an error even if runstop is `True`. + +* `~run_stop_topic` (`String`, default: `None`) + + Subscribe this topic if this value is specified. + +* `~run_stop_condition` (`String`, default: `m.data == True`) + + Returning bool value condition using the given Python expression. + The Python expression can access any of the Python builtins plus: + ``topic`` (the topic of the message), ``m`` (the message) and ``t`` (time of message). + + For example, ``~run_stop_topic`` is ``robot_state (fetch_driver_msgs/RobotState)`` and if you want to check whether a runstop is a pressed, you can do the following. + + ```bash + run_stop_condition: "m.runstopped is True" + ``` + +* `~ignore_time_after_runstop_is_enabled` (`Float`, default: `0.0`) + + Time to ignore diagnostics after runstop is enabled. + +* `~ignore_time_after_runstop_is_disabled` (`Float`, default: `0.0`) + + Time to ignore diagnostics after runstop is disabled. + +- `~blacklist` (`Yaml`, required) + + User must always specify `name`. You can specify `message` as an option. + + These values are matched using python regular expressions. + + It is something like below: + + ``` + + + + + run_stop_topic: robot_state + run_stop_condition: "m.runstopped is True" + seconds_to_start_speaking: 30 + blacklist: + - "/CPU/CPU Usage/my_machine CPU Usage" + - "/SoundPlay/sound_play: Node State" + - "/Other/jsk_joy_node: Joystick Driver Status" + - "/Other/Combined Gyro" + - "/Other/l515_head l515_head_realsense2_camera_color: Frequency Status" + - "/Other/l515_head l515_head_realsense2_camera_confidence: Frequency Status" + - "/Other/l515_head l515_head_realsense2_camera_depth: Frequency Status" + - "/Other/l515_head l515_head_realsense2_camera_infra: Frequency Status" + - "/Other/ukf_se: Filter diagnostic updater" + - "/Peripherals/PS3 Controller" + - "/Sensors/IMU/IMU 1 Gyro" + - "/Sensors/IMU/IMU 2 Gyro" + run_stop_blacklist: + - "\\w*_(mcb|breaker)" + - "/Motor Control Boards/.*" + - "/Breakers/.*" + + + ``` + +- `~run_stop_blacklist` (`Yaml`, optional) + + This is valid when run_stop is `True`. Blacklist at run_stop. + + +## Usage + +Listen warnings from diagnostics. + +```bash +roslaunch jsk_tools sample_audible_warning.launch +``` + +![audible_warning](./images/audible_warning.jpg) diff --git a/doc/jsk_tools/scripts/images/audible_warning.jpg b/doc/jsk_tools/scripts/images/audible_warning.jpg new file mode 100644 index 000000000..5d3ed1f7d Binary files /dev/null and b/doc/jsk_tools/scripts/images/audible_warning.jpg differ diff --git a/jsk_tools/CMakeLists.txt b/jsk_tools/CMakeLists.txt index c1a9e6779..451998060 100644 --- a/jsk_tools/CMakeLists.txt +++ b/jsk_tools/CMakeLists.txt @@ -1,11 +1,18 @@ cmake_minimum_required(VERSION 2.8.3) project(jsk_tools) -find_package(catkin REQUIRED) +find_package(catkin REQUIRED + COMPONENTS + dynamic_reconfigure) catkin_python_setup() set(ROS_BUILD_TYPE RelWithDebInfo) + +generate_dynamic_reconfigure_options( + cfg/AudibleWarning.cfg +) + catkin_package( - CATKIN_DEPENDS # + CATKIN_DEPENDS dynamic_reconfigure # LIBRARIES # INCLUDE_DIRS # DEPENDS # @@ -27,6 +34,7 @@ endforeach(exec) if (CATKIN_ENABLE_TESTING) find_package(roslint REQUIRED) + roslint_python(node_scripts/audible_warning.py) roslint_python(src/jsk_tools/cltool.py) roslint_python(src/test_topic_published.py) roslint_python(src/test_rosparam_set.py) @@ -49,6 +57,7 @@ if (CATKIN_ENABLE_TESTING) # https://github.com/jsk-ros-pkg/jsk_common/pull/1293#issuecomment-164158260 jsk_tools_add_rostest(test/test_rosparam_set.test) endif() + jsk_tools_add_rostest(test/test_audible_warning.test) jsk_tools_add_rostest(test/test_stdout.test) jsk_tools_add_rostest(test/test_rostopic_host_sanity.test) jsk_tools_add_rostest(test/test_sanity_diagnostics.test) @@ -74,7 +83,7 @@ install(DIRECTORY test USE_SOURCE_PERMISSIONS PATTERN "*.test" EXCLUDE ) -install(DIRECTORY src dot-files cmake launch sample +install(DIRECTORY src dot-files cmake launch sample node_scripts DESTINATION ${CATKIN_PACKAGE_SHARE_DESTINATION} USE_SOURCE_PERMISSIONS ) diff --git a/jsk_tools/cfg/AudibleWarning.cfg b/jsk_tools/cfg/AudibleWarning.cfg new file mode 100755 index 000000000..705682213 --- /dev/null +++ b/jsk_tools/cfg/AudibleWarning.cfg @@ -0,0 +1,26 @@ +#! /usr/bin/env python + + +from dynamic_reconfigure.parameter_generator_catkin import * + +PACKAGE = 'jsk_tools' + +gen = ParameterGenerator() +speak_group = gen.add_group("speak flag") +speak_group.add("enable", bool_t, 0, "Flag of speak", True) +speak_group.add("speak_ok", bool_t, 0, "Speak ok level diagnostics", False) +speak_group.add("speak_stale", bool_t, 0, "Speak stale level diagnostics", True) +speak_group.add("speak_warn", bool_t, 0, "Speak warn level diagnostics", True) +speak_group.add("speak_error", bool_t, 0, "Speak error level diagnostics", True) +speak_group.add("speak_when_runstopped", bool_t, 0, "Speak when runstop is enabled", True) + +gen.add("volume", double_t, 0, "Volume of sound.", 1.0, 0.0, 1.0) +gen.add("speak_interval", double_t, 0, "Volume of sound.", 120.0, 0.0, 3600.0) +gen.add("ignore_time_after_runstop_is_enabled", double_t, 0, + "Time to ignore diagnostics after runstop is enabled.", + 0.0, 0.0, 3600.0) +gen.add("ignore_time_after_runstop_is_disabled", double_t, 0, + "Time to ignore diagnostics after runstop is disabled.", + 0.0, 0.0, 3600.0) + +exit(gen.generate(PACKAGE, PACKAGE, "AudibleWarning")) diff --git a/jsk_tools/node_scripts/audible_warning.py b/jsk_tools/node_scripts/audible_warning.py new file mode 100755 index 000000000..f1bd39845 --- /dev/null +++ b/jsk_tools/node_scripts/audible_warning.py @@ -0,0 +1,334 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- + +from collections import defaultdict +import heapq +from importlib import import_module +import re +from threading import Event +from threading import Lock +from threading import Thread + +import actionlib +from diagnostic_msgs.msg import DiagnosticArray +from diagnostic_msgs.msg import DiagnosticStatus +from dynamic_reconfigure.server import Server +import std_msgs.msg +import rospy +from sound_play.msg import SoundRequest +from sound_play.msg import SoundRequestAction +from sound_play.msg import SoundRequestGoal + +from jsk_tools.cfg import AudibleWarningConfig as Config +from jsk_tools.diagnostics_utils import diagnostics_level_to_str +from jsk_tools.diagnostics_utils import filter_diagnostics_status_list +from jsk_tools.diagnostics_utils import is_leaf +from jsk_tools.inflection_utils import camel_to_snake +from jsk_tools.string_utils import multiple_whitespace_to_one + + +def expr_eval(expr): + def eval_fn(topic, m, t): + return eval(expr) + return eval_fn + + +class SpeakThread(Thread): + + def __init__(self, rate=1.0, wait=True, + language='', + volume=1.0, + speak_interval=0, + wait_speak_duration_time=10, + diagnostics_level_list=None): + super(SpeakThread, self).__init__() + self.wait_speak_duration_time = wait_speak_duration_time + self.event = Event() + self.rate = rate + self.wait = wait + self.volume = volume + self.lock = Lock() + self.status_list = [] + self.speak_interval = speak_interval + self.diagnostics_level_list = diagnostics_level_list or [] + tm = rospy.Time.now().to_sec() \ + - speak_interval + self.previous_spoken_time = defaultdict(lambda tm=tm: tm) + self.speak_flag = True + self.language = language + + self.pub_original_text = rospy.Publisher('~output/original_text', + std_msgs.msg.String, + queue_size=1) + self.pub_speak_text = rospy.Publisher('~output/text', + std_msgs.msg.String, + queue_size=1) + + self.talk = actionlib.SimpleActionClient( + "/robotsound", SoundRequestAction) + self.talk.wait_for_server() + + def stop(self): + self.event.set() + + def set_diagnostics_level_list(self, level_list): + self.diagnostics_level_list = level_list + + def set_speak_flag(self, flag): + if flag is True and self.speak_flag is False: + # clear queue before start speaking. + with self.lock: + self.status_list = [] + self.speak_flag = flag + if self.speak_flag is True: + rospy.loginfo('audible warning is enabled. speak [{}] levels' + .format(', '.join(map(diagnostics_level_to_str, + self.diagnostics_level_list)))) + else: + rospy.loginfo('audible warning is disabled.') + + def set_volume(self, volume): + volume = min(max(0.0, volume), 1.0) + if self.volume != volume: + self.volume = volume + rospy.loginfo("audible warning's volume was set to {}".format( + self.volume)) + + def set_speak_interval(self, interval): + interval = max(0.0, interval) + if self.speak_interval != interval: + self.speak_interval = interval + rospy.loginfo("audible warning's speak interval was set to {}" + .format(self.speak_interval)) + + def add(self, status_list): + with self.lock: + for status in status_list: + if is_leaf(status.name) is False: + continue + if rospy.Time.now().to_sec() \ + - self.previous_spoken_time[status.name] \ + < self.speak_interval: + continue + heapq.heappush( + self.status_list, + (rospy.Time.now().to_sec(), status)) + + def pop(self): + with self.lock: + while len(self.status_list) > 0: + _, status = heapq.heappop(self.status_list) + if is_leaf(status.name) is False: + continue + if rospy.Time.now().to_sec() \ + - self.previous_spoken_time[status.name] \ + < self.speak_interval: + continue + return status + return None + + def run(self): + while not self.event.wait(self.rate): + e = self.pop() + if e: + if self.speak_flag is False: + continue + if e.level not in self.diagnostics_level_list: + continue + + if e.level == DiagnosticStatus.OK: + prefix = 'ok.' + elif e.level == DiagnosticStatus.WARN: + prefix = 'warning.' + elif e.level == DiagnosticStatus.ERROR: + prefix = 'error.' + elif e.level == DiagnosticStatus.STALE: + prefix = 'stale.' + else: + prefix = 'ok.' + sentence = prefix + e.name + ' ' + e.message + sentence = camel_to_snake(sentence) + sentence = sentence.replace('/', ' ') + sentence = sentence.replace('_', ' ') + sentence = sentence.replace(':', ' colon ') + sentence = multiple_whitespace_to_one(sentence) + rospy.loginfo('audible warning error name "{}"'.format(e.name)) + rospy.loginfo("audible warning talking: %s" % sentence) + self.pub_original_text.publish(prefix + ' ' + e.name + ' ' + e.message) + self.pub_speak_text.publish( + "audible warning talking: %s" % sentence) + + goal = SoundRequestGoal() + goal.sound_request.sound = SoundRequest.SAY + goal.sound_request.command = SoundRequest.PLAY_ONCE + goal.sound_request.arg = sentence + goal.sound_request.arg2 = self.language + if hasattr(goal.sound_request, 'volume'): + goal.sound_request.volume = self.volume + + self.previous_spoken_time[e.name] = rospy.Time.now().to_sec() + self.talk.send_goal(goal) + if self.wait: + self.talk.wait_for_result( + rospy.Duration(self.wait_speak_duration_time)) + + +class AudibleWarning(object): + + def __init__(self): + speak_rate = rospy.get_param("~speak_rate", 1.0 / 100.0) + wait_speak = rospy.get_param("~wait_speak", True) + language = rospy.get_param('~language', '') + seconds_to_start_speaking = rospy.get_param( + '~seconds_to_start_speaking', 0) + wait_speak_duration_time = rospy.get_param( + '~wait_speak_duration_time', 30.0) + + # Wait until seconds_to_start_speaking the time has passed. + self.run_stop_enabled_time = None + self.run_stop_disabled_time = None + rate = rospy.Rate(10) + start_time = rospy.Time.now() + while not rospy.is_shutdown() \ + and (rospy.Time.now() - start_time).to_sec() \ + < seconds_to_start_speaking: + rate.sleep() + + blacklist = rospy.get_param("~blacklist", []) + self.blacklist_names = [] + self.blacklist_messages = [] + for bl in blacklist: + if 'name' not in bl: + name = re.compile(r'.') + else: + name = re.compile(bl['name']) + self.blacklist_names.append(name) + if 'message' not in bl: + message = re.compile(r'.') + else: + message = re.compile(bl['message']) + self.blacklist_messages.append(message) + self.speak_thread = SpeakThread( + speak_rate, wait_speak, + language, + wait_speak_duration_time=wait_speak_duration_time) + self.srv = Server(Config, self.config_callback) + + # run-stop + self.run_stop = False + self.run_stop_topic = rospy.get_param('~run_stop_topic', None) + if self.run_stop_topic: + run_stop_condition = rospy.get_param( + '~run_stop_condition', 'm.data == True') + run_stop_blacklist = rospy.get_param( + '~run_stop_blacklist', []) + self.run_stop_blacklist_names = [] + self.run_stop_blacklist_messages = [] + for bl in run_stop_blacklist: + if 'name' not in bl: + name = re.compile(r'.') + else: + name = re.compile(bl['name']) + self.run_stop_blacklist_names.append(name) + if 'message' not in bl: + message = re.compile(r'.') + else: + message = re.compile(bl['message']) + self.run_stop_blacklist_messages.append(message) + self.run_stop_condition = expr_eval(run_stop_condition) + self.run_stop_sub = rospy.Subscriber( + self.run_stop_topic, + rospy.AnyMsg, + + callback=self.run_stop_callback, + queue_size=1) + + # diag + self.sub_diag = rospy.Subscriber( + "/diagnostics_agg", DiagnosticArray, + self.diag_cb, queue_size=1) + self.speak_thread.start() + + def config_callback(self, config, level): + level_list = [] + if config.speak_ok: + level_list.append(DiagnosticStatus.OK) + if config.speak_warn: + level_list.append(DiagnosticStatus.WARN) + if config.speak_error: + level_list.append(DiagnosticStatus.ERROR) + if config.speak_stale: + level_list.append(DiagnosticStatus.STALE) + self.speak_thread.set_diagnostics_level_list(level_list) + self.speak_thread.set_speak_flag(config.enable) + self.speak_thread.set_volume(config.volume) + self.speak_thread.set_speak_interval(config.speak_interval) + self.ignore_time_after_runstop_is_enabled = \ + config.ignore_time_after_runstop_is_enabled + self.ignore_time_after_runstop_is_disabled = \ + config.ignore_time_after_runstop_is_disabled + self.speak_when_runstopped = config.speak_when_runstopped + return config + + def run_stop_callback(self, msg): + if isinstance(msg, rospy.msg.AnyMsg): + package, msg_type = msg._connection_header['type'].split('/') + ros_pkg = package + '.msg' + msg_class = getattr(import_module(ros_pkg), msg_type) + self.run_stop_sub.unregister() + deserialized_sub = rospy.Subscriber( + self.run_stop_topic, msg_class, self.run_stop_callback) + self.run_stop_sub = deserialized_sub + msg = msg_class().deserialize(msg._buff) + tm = rospy.Time.now() + run_stop = self.run_stop_condition( + self.run_stop_topic, msg, tm) + if run_stop != self.run_stop: + if run_stop is True: + self.run_stop_enabled_time = rospy.Time.now() + rospy.loginfo('Audible Warning: Runstop is enabled.') + else: + self.run_stop_disabled_time = rospy.Time.now() + rospy.loginfo('Audible Warning: Runstop is disabled.') + self.run_stop = run_stop + + def on_shutdown(self): + self.speak_thread.stop() + self.speak_thread.join() + + def diag_cb(self, msg): + target_status_list = msg.status + + if self.ignore_time_after_runstop_is_enabled > 0.0: + if self.run_stop_enabled_time is not None \ + and ((rospy.Time.now() - + self.run_stop_enabled_time).to_sec() < + self.ignore_time_after_runstop_is_enabled): + return + if self.ignore_time_after_runstop_is_disabled > 0.0: + if self.run_stop_disabled_time is not None \ + and ((rospy.Time.now() - + self.run_stop_disabled_time).to_sec() < + self.ignore_time_after_runstop_is_disabled): + return + + if self.run_stop: + if self.speak_when_runstopped is False: + rospy.logdebug('RUN STOP is pressed. Do not speak warning.') + return + + target_status_list = filter_diagnostics_status_list( + target_status_list, + self.run_stop_blacklist_names, + self.run_stop_blacklist_messages) + + target_status_list = filter_diagnostics_status_list( + target_status_list, self.blacklist_names, self.blacklist_messages) + self.speak_thread.add(target_status_list) + + +if __name__ == '__main__': + rospy.init_node("audible_warning") + aw = AudibleWarning() # NOQA + rospy.on_shutdown(aw.on_shutdown) + rospy.spin() diff --git a/jsk_tools/package.xml b/jsk_tools/package.xml index ef0a9d455..082dc1b2b 100644 --- a/jsk_tools/package.xml +++ b/jsk_tools/package.xml @@ -15,12 +15,14 @@ catkin + dynamic_reconfigure git rosgraph_msgs cv_bridge diagnostic_aggregator diagnostic_msgs diagnostic_updater + dynamic_reconfigure python-percol python-colorama python3-colorama diff --git a/jsk_tools/sample/config/sample_audible_warning_diagnostics_analyzer.yaml b/jsk_tools/sample/config/sample_audible_warning_diagnostics_analyzer.yaml new file mode 100644 index 000000000..c6c0feb93 --- /dev/null +++ b/jsk_tools/sample/config/sample_audible_warning_diagnostics_analyzer.yaml @@ -0,0 +1,15 @@ +analyzers: + cameras: + type: diagnostic_aggregator/AnalyzerGroup + path: Sensors + analyzers: + pseudo_diagnostics1: + type: diagnostic_aggregator/GenericAnalyzer + path: sensor1 + find_and_remove_prefix: 'pseudo_diagnostics: ' + num_items: 1 + pseudo_diagnostics2: + type: diagnostic_aggregator/GenericAnalyzer + path: sensor2 + find_and_remove_prefix: 'pseudo_diagnostics: ' + num_items: 1 diff --git a/jsk_tools/sample/pseudo_diagnostics.py b/jsk_tools/sample/pseudo_diagnostics.py new file mode 100755 index 000000000..92f0b4d4c --- /dev/null +++ b/jsk_tools/sample/pseudo_diagnostics.py @@ -0,0 +1,56 @@ +#!/usr/bin/env python + +import diagnostic_msgs +import diagnostic_updater +from distutils.version import LooseVersion as version +import pkg_resources +import rospy + + +def change_diagnostics(stat): + current_state = True + + def _change_diagnostics(stat): + if current_state: + stat.summary(diagnostic_msgs.msg.DiagnosticStatus.OK, + "Sensor is OK") + else: + stat.summary(diagnostic_msgs.msg.DiagnosticStatus.ERROR, + "Sensor is not OK.") + current_state = not current_state + stat.add("Current State", current_state) + return stat + return _change_diagnostics + + +class PseudoDiagnostics(object): + + def __init__(self, sensor_name, duration_time=5): + self.updater = diagnostic_updater.Updater() + self.updater.setHardwareID(sensor_name) + self.updater.add("pseudo_diagnostics", change_diagnostics) + kwargs = dict( + period=rospy.Duration(duration_time), + callback=self.update_diagnostics, + oneshot=False, + ) + if version(pkg_resources.get_distribution('rospy').version) \ + >= version('1.12.0'): + # on >=kinetic, it raises ROSTimeMovedBackwardsException + # when we use rosbag play --loop. + kwargs['reset'] = True + self.timer = rospy.Timer(**kwargs) + + def update_diagnostics(self, event): + self.updater.update() + + +if __name__ == '__main__': + rospy.init_node('pseudo_diagnostics') + pusedo_diagnotics = PseudoDiagnostics( + sensor_name='sensor1', + duration_time=5) + pusedo_diagnotics = PseudoDiagnostics( + sensor_name='sensor2', + duration_time=10) + rospy.spin() diff --git a/jsk_tools/sample/pseudo_runstop_publisher.py b/jsk_tools/sample/pseudo_runstop_publisher.py new file mode 100755 index 000000000..7091fc95f --- /dev/null +++ b/jsk_tools/sample/pseudo_runstop_publisher.py @@ -0,0 +1,41 @@ +#!/usr/bin/env python + +from distutils.version import LooseVersion as version + +import pkg_resources +import rospy +import std_msgs.msg + + +class PseudoRunstopPublisher(object): + + def __init__(self): + duration_time = rospy.get_param('~duration_time', 30) + self.runstop = True + self.pub = rospy.Publisher('~runstop', std_msgs.msg.Bool, + queue_size=1) + kwargs = dict( + period=rospy.Duration(duration_time), + callback=self.publish_callback, + oneshot=False, + ) + if version(pkg_resources.get_distribution('rospy').version) \ + >= version('1.12.0'): + # on >=kinetic, it raises ROSTimeMovedBackwardsException + # when we use rosbag play --loop. + kwargs['reset'] = True + self.timer = rospy.Timer(**kwargs) + + def publish_callback(self, event): + if self.runstop is True: + rospy.loginfo('Runstop is enabled.') + else: + rospy.loginfo('Runstop is disabled.') + self.pub.publish(self.runstop) + self.runstop = not self.runstop + + +if __name__ == '__main__': + rospy.init_node('pseudo_runstop_publisher') + pub = PseudoRunstopPublisher() + rospy.spin() diff --git a/jsk_tools/sample/sample_audible_warning.launch b/jsk_tools/sample/sample_audible_warning.launch new file mode 100644 index 000000000..f9fb14ed0 --- /dev/null +++ b/jsk_tools/sample/sample_audible_warning.launch @@ -0,0 +1,53 @@ + + + + + + + + + + + + duration_time: 30 + + + + + + + + + + + + + + + + run_stop_topic: /pseudo_runstop_publisher/runstop + speak_when_runstopped: true + speak_interval: 0.0 + seconds_to_start_speaking: 5 + ignore_time_after_runstop_is_enabled: 10.0 + ignore_time_after_runstop_is_disabled: 10.0 + + + + + + + + + diff --git a/jsk_tools/src/jsk_tools/clients/__init__.py b/jsk_tools/src/jsk_tools/clients/__init__.py new file mode 100644 index 000000000..91a96e71c --- /dev/null +++ b/jsk_tools/src/jsk_tools/clients/__init__.py @@ -0,0 +1 @@ +from .audible_warninig_client import AudibleWarningReconfigureClient diff --git a/jsk_tools/src/jsk_tools/clients/audible_warninig_client.py b/jsk_tools/src/jsk_tools/clients/audible_warninig_client.py new file mode 100644 index 000000000..0bc575505 --- /dev/null +++ b/jsk_tools/src/jsk_tools/clients/audible_warninig_client.py @@ -0,0 +1,18 @@ +from dynamic_reconfigure.client import Client + + +class AudibleWarningReconfigureClient(object): + + def __init__(self, client_name='audible_warning'): + super(AudibleWarningReconfigureClient).__init__() + self.client_name = client_name + self.client = Client(client_name, timeout=3) + + def reconfigure(self, **kwargs): + return self.client.update_configuration(kwargs) + + def enable(self): + return self.client.update_configuration({'enable': True}) + + def disable(self): + return self.client.update_configuration({'enable': False}) diff --git a/jsk_tools/src/jsk_tools/diagnostics_utils.py b/jsk_tools/src/jsk_tools/diagnostics_utils.py new file mode 100644 index 000000000..a65c506bc --- /dev/null +++ b/jsk_tools/src/jsk_tools/diagnostics_utils.py @@ -0,0 +1,92 @@ +try: + from itertools import zip_longest +except: + from itertools import izip_longest as zip_longest +import re + +from diagnostic_msgs.msg import DiagnosticStatus + + +cached_paths = {} +cached_result = {} + + +def is_leaf(original_path): + if hasattr(cached_result, original_path): + return cached_result[original_path] + + path = original_path[1:].split('/') + + def _is_leaf(cached, path): + if not isinstance(path, list): + raise ValueError + if len(path) == 0: + raise ValueError + if len(path) == 1: + if path[0] in cached: + return False + else: + return True + + parent = path[0] + child = path[1:] + if parent not in cached: + cached[parent] = {} + return _is_leaf(cached[parent], child) + + result = _is_leaf(cached_paths, path) + if result is False: + cached_result[original_path] = result + return result + + +def filter_diagnostics_status_list(status_list, blacklist, + blacklist_messages=None): + """Filter list of DiagnosticStatus. + + Parameters + ---------- + status_list : List[DiagnosticStatus] + List of DiagnosticStatus. + blacklist : List[str] or List[re.Pattern] + List of blacklist. + The path of the name contained in this list is ignored. + blacklist_messages : List[str] or List[re.Pattern] + List of blacklist for message. + This has a one-to-one correspondence with blacklist. + + Returns + ------- + filtered_status : List[DiagnosticStatus] + List of filtered diagnostics status. + """ + blacklist_messages = blacklist_messages or [] + filtered_status = [] + for s in status_list: + ns = s.name + if is_leaf(ns) is False: + continue + matched = False + for bn, message in zip_longest( + blacklist, blacklist_messages): + if re.match(bn, ns): + if message is None or re.match(message, s.message): + matched = True + break + if matched is True: + continue + filtered_status.append(s) + return filtered_status + + +def diagnostics_level_to_str(level): + if level == DiagnosticStatus.OK: + return 'OK' + elif level == DiagnosticStatus.WARN: + return 'WARN' + elif level == DiagnosticStatus.ERROR: + return 'ERROR' + elif level == DiagnosticStatus.STALE: + return 'STALE' + else: + raise ValueError('Not valid level {}'.format(level)) diff --git a/jsk_tools/src/jsk_tools/inflection_utils.py b/jsk_tools/src/jsk_tools/inflection_utils.py new file mode 100644 index 000000000..455869920 --- /dev/null +++ b/jsk_tools/src/jsk_tools/inflection_utils.py @@ -0,0 +1,15 @@ +import re + + +def camel_to_snake(word): + """Convert Camel case to snake case. + + Examples + -------- + >>> camel_to_snake('clusterPointIndices') + 'cluster_point_indices' + """ + word = re.sub(r"([A-Z]+)([A-Z][a-z])", r'\1_\2', word) + word = re.sub(r"([a-z\d])([A-Z])", r'\1_\2', word) + word = word.replace("-", "_") + return word.lower() diff --git a/jsk_tools/src/jsk_tools/string_utils.py b/jsk_tools/src/jsk_tools/string_utils.py new file mode 100644 index 000000000..fe9459ce5 --- /dev/null +++ b/jsk_tools/src/jsk_tools/string_utils.py @@ -0,0 +1,2 @@ +def multiple_whitespace_to_one(text): + return ' '.join(text.split()) diff --git a/jsk_tools/test/test_audible_warning.test b/jsk_tools/test/test_audible_warning.test new file mode 100644 index 000000000..9efa2e8b3 --- /dev/null +++ b/jsk_tools/test/test_audible_warning.test @@ -0,0 +1,17 @@ + + + + + + + + + + topic_0: /sound_play/goal + timeout_0: 10 + + + +