topic_helpers.py
Go to the documentation of this file.
00001 # Copyright (c) 2011, Dorian Scholz, TU Darmstadt
00002 # All rights reserved.
00003 #
00004 # Redistribution and use in source and binary forms, with or without
00005 # modification, are permitted provided that the following conditions
00006 # are met:
00007 #
00008 #   * Redistributions of source code must retain the above copyright
00009 #     notice, this list of conditions and the following disclaimer.
00010 #   * Redistributions in binary form must reproduce the above
00011 #     copyright notice, this list of conditions and the following
00012 #     disclaimer in the documentation and/or other materials provided
00013 #     with the distribution.
00014 #   * Neither the name of the TU Darmstadt nor the names of its
00015 #     contributors may be used to endorse or promote products derived
00016 #     from this software without specific prior written permission.
00017 #
00018 # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
00019 # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
00020 # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
00021 # FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
00022 # COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
00023 # INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
00024 # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
00025 # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
00026 # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
00027 # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
00028 # ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
00029 # POSSIBILITY OF SUCH DAMAGE.
00030 
00031 import roslib
00032 roslib.load_manifest('rqt_py_common')
00033 import roslib.msgs
00034 from rostopic import get_topic_type
00035 from python_qt_binding.QtCore import qDebug
00036 
00037 
00038 def get_field_type(topic_name):
00039     """
00040     Get the Python type of a specific field in the given registered topic.
00041     If the field is an array, the type of the array's values are returned and the is_array flag is set to True.
00042     This is a static type check, so it works for unpublished topics and with empty arrays.
00043 
00044     :param topic_name: name of field of a registered topic, ``str``, i.e. '/rosout/file'
00045     :returns: field_type, is_array
00046     """
00047     # get topic_type and message_evaluator
00048     topic_type, real_topic_name, _ = get_topic_type(topic_name)
00049     if topic_type is None:
00050         #qDebug('topic_helpers.get_field_type(%s): get_topic_type failed' % (topic_name))
00051         return None, False
00052 
00053     message_class = roslib.message.get_message_class(topic_type)
00054     if message_class is None:
00055         qDebug('topic_helpers.get_field_type(%s): get_message_class(%s) failed' % (topic_name, topic_type))
00056         return None, False
00057 
00058     slot_path = topic_name[len(real_topic_name):]
00059     return get_slot_type(message_class, slot_path)
00060 
00061 
00062 def get_slot_type(message_class, slot_path):
00063     """
00064     Get the Python type of a specific slot in the given message class.
00065     If the field is an array, the type of the array's values are returned and the is_array flag is set to True.
00066     This is a static type check, so it works for unpublished topics and with empty arrays.
00067 
00068     :param message_class: message class type, ``type``, usually inherits from genpy.message.Message
00069     :param slot_path: path to the slot inside the message class, ``str``, i.e. 'header/seq'
00070     :returns: field_type, is_array
00071     """
00072     is_array = False
00073     fields = [f for f in slot_path.split('/') if f]
00074     for field_name in fields:
00075         try:
00076             field_name, _, field_index = roslib.msgs.parse_type(field_name)
00077         except roslib.msgs.MsgSpecException:
00078             return None, False
00079         if field_name not in getattr(message_class, '__slots__', []):
00080             #qDebug('topic_helpers.get_slot_type(%s, %s): field not found: %s' % (message_class, slot_path, field_name))
00081             return None, False
00082         slot_type = message_class._slot_types[message_class.__slots__.index(field_name)]
00083         slot_type, slot_is_array, _ = roslib.msgs.parse_type(slot_type)
00084         is_array = slot_is_array and field_index is None
00085         if roslib.msgs.is_valid_constant_type(slot_type):
00086             if slot_type == 'string':
00087                 message_class = str
00088             elif slot_type == 'bool':
00089                 message_class = bool
00090             else:
00091                 message_class = type(roslib.msgs._convert_val(slot_type, 0))
00092         else:
00093             message_class = roslib.message.get_message_class(slot_type)
00094     return message_class, is_array
00095 
00096 
00097 def is_slot_numeric(topic_name):
00098     """
00099     Check is a slot in the given topic is numeric, or an array of numeric values.
00100     This is a static type check, so it works for unpublished topics and with empty arrays.
00101 
00102     :param topic_name: name of field of a registered topic, ``str``, i.e. '/rosout/file'
00103     :returns: is_numeric, is_array, description
00104     """
00105     field_type, is_array = get_field_type(topic_name)
00106     if field_type in (int, float):
00107         if is_array:
00108             message = 'topic "%s" is numeric array: %s[]' % (topic_name, field_type)
00109         else:
00110             message = 'topic "%s" is numeric: %s' % (topic_name, field_type)
00111         return True, is_array, message
00112 
00113     return False, is_array, 'topic "%s" is NOT numeric: %s' % (topic_name, field_type)


rqt_py_common
Author(s): Dorian Scholz
autogenerated on Fri Jan 3 2014 11:54:45