Skip to content

Commit

Permalink
Handle ROS 2 types properly (RobotWebTools#883)
Browse files Browse the repository at this point in the history
  • Loading branch information
scottbell authored Oct 27, 2023
1 parent 501a926 commit 910163b
Show file tree
Hide file tree
Showing 3 changed files with 132 additions and 23 deletions.
1 change: 1 addition & 0 deletions rosapi/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -24,4 +24,5 @@ install(
if(BUILD_TESTING)
find_package(ament_cmake_pytest REQUIRED)
ament_add_pytest_test(${PROJECT_NAME}_test_stringify_field_types test/test_stringify_field_types.py)
ament_add_pytest_test(${PROJECT_NAME}_test_typedefs test/test_typedefs.py)
endif()
108 changes: 85 additions & 23 deletions rosapi/src/rosapi/objectutils.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,13 +32,16 @@
# POSSIBILITY OF SUCH DAMAGE.

import inspect
import logging
import re

from rosapi.stringify_field_types import stringify_field_types
from rosbridge_library.internal import ros_loader

# Keep track of atomic types and special types
atomics = [
"bool",
"boolean",
"byte",
"int8",
"uint8",
Expand All @@ -48,9 +51,12 @@
"uint32",
"int64",
"uint64",
"float",
"float32",
"float64",
"double",
"string",
"octet",
]
specials = ["time", "duration"]

Expand All @@ -65,6 +71,12 @@ def get_typedef(type):
- string[] constnames
- string[] constvalues
get_typedef will return a typedef dict for the specified message type"""

# Check if the type string indicates a sequence (array) type
if matches := re.findall("sequence<([^<]+)>", type):
# Extract the inner type and continue processing
type = matches[0]

if type in atomics:
# Atomics don't get a typedef
return None
Expand All @@ -74,8 +86,13 @@ def get_typedef(type):
return _get_special_typedef(type)

# Fetch an instance and return its typedef
instance = ros_loader.get_message_instance(type)
return _get_typedef(instance)
try:
instance = ros_loader.get_message_instance(type)
type_def = _get_typedef(instance)
return type_def
except (ros_loader.InvalidModuleException, ros_loader.InvalidClassException) as e:
logging.error(f"An error occurred trying to get the type definition for {type}: {e}")
return None


def get_service_request_typedef(servicetype):
Expand Down Expand Up @@ -128,27 +145,63 @@ def get_typedef_full_text(ty):

def _get_typedef(instance):
"""Gets a typedef dict for the specified instance"""
if (
if _valid_instance(instance):
fieldnames, fieldtypes, fieldarraylen, examples = _handle_array_information(instance)
constnames, constvalues = _handle_constant_information(instance)
typedef = _build_typedef_dictionary(
instance, fieldnames, fieldtypes, fieldarraylen, examples, constnames, constvalues
)
return typedef


def _valid_instance(instance):
"""Check if instance is valid i.e.,
not None, has __slots__ and _fields_and_field_types attributes"""
return not (
instance is None
or not hasattr(instance, "__slots__")
or not hasattr(instance, "_fields_and_field_types")
):
return None
)


def _handle_array_information(instance):
"""Handles extraction of array information including field names, types,
lengths and examples"""
fieldnames = []
fieldtypes = []
fieldarraylen = []
examples = []
constnames = []
constvalues = []
for i in range(len(instance.__slots__)):
# Pull out the name
name = instance.__slots__[i]
fieldnames.append(name)

# Pull out the type and determine whether it's an array
field_type = instance._fields_and_field_types[name[1:]] # Remove trailing underscore.
arraylen = -1
field_type, arraylen = _handle_type_and_array_len(instance, name)
fieldarraylen.append(arraylen)

field_instance = getattr(instance, name)
fieldtypes.append(_type_name(field_type, field_instance))

example = _handle_example(arraylen, field_type, field_instance)
examples.append(str(example))

return fieldnames, fieldtypes, fieldarraylen, examples


def _handle_type_and_array_len(instance, name):
"""Extracts field type and determines its length if it's an array"""

# Get original field type using instance's _fields_and_field_types property
field_type = instance._fields_and_field_types[name[1:]]

# Initialize arraylen
arraylen = -1

# If field_type is a sequence, update the `field_type` variable.
if matches := re.findall("sequence<([^<]+)>", field_type):
# Extract the inner type and continue processing
field_type = matches[0]
arraylen = 0
else:
if field_type[-1:] == "]":
if field_type[-2:-1] == "[":
arraylen = 0
Expand All @@ -157,21 +210,25 @@ def _get_typedef(instance):
split = field_type.find("[")
arraylen = int(field_type[split + 1 : -1])
field_type = field_type[:split]
fieldarraylen.append(arraylen)

# Get the fully qualified type
field_instance = getattr(instance, name)
fieldtypes.append(_type_name(field_type, field_instance))
return field_type, arraylen

# Set the example as appropriate

def _handle_example(arraylen, field_type, field_instance):
"""Determines the example of a field instance, whether it's an array or atomic type"""
if arraylen >= 0:
example = []
elif field_type not in atomics:
example = {}
else:
example = field_instance
if arraylen >= 0:
example = []
elif field_type not in atomics:
example = {}
examples.append(str(example))
return example


# Add pseudo constants names and values filtering members
def _handle_constant_information(instance):
"""Handles extraction of constants information including constant names and values"""
constnames = []
constvalues = []
attributes = inspect.getmembers(instance)
for attribute in attributes:
if (
Expand All @@ -181,7 +238,13 @@ def _get_typedef(instance):
):
constnames.append(str(attribute[0]))
constvalues.append(str(attribute[1]))
return constnames, constvalues


def _build_typedef_dictionary(
instance, fieldnames, fieldtypes, fieldarraylen, examples, constnames, constvalues
):
"""Builds the typedef dictionary from multiple inputs collected from instance"""
typedef = {
"type": _type_name_from_instance(instance),
"fieldnames": fieldnames,
Expand All @@ -191,7 +254,6 @@ def _get_typedef(instance):
"constnames": constnames,
"constvalues": constvalues,
}

return typedef


Expand Down
46 changes: 46 additions & 0 deletions rosapi/test/test_typedefs.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
#!/usr/bin/env python
import unittest

import rosapi.objectutils as objectutils

# Globally defined ros_loader, used inside the setUp and teardown functions
ros_loader = None


class TestUtils(unittest.TestCase):
def setUp(self):
global ros_loader
self.original_ros_loader = ros_loader
ros_loader = self._mock_get_message_instance("default")

def tearDown(self):
global ros_loader
ros_loader = self.original_ros_loader

def _mock_get_message_instance(self, type):
class MockInstance(object):
__slots__ = ["_" + type]
_fields_and_field_types = {type: type}

return MockInstance()

def test_get_typedef_for_atomic_types(self):
# Test for boolean type
actual_typedef = objectutils.get_typedef("boolean")
# should be None for an atomic
self.assertEqual(actual_typedef, None)

# Test for float type
actual_typedef = objectutils.get_typedef("float")
# should be None for an atomic
self.assertEqual(actual_typedef, None)

def test_handle_sequences(self):
# Test for boolean sequence type
actual_typedef = objectutils.get_typedef("sequence<boolean>")
# should be None for an atomic
self.assertEqual(actual_typedef, None)


if __name__ == "__main__":
unittest.main()

0 comments on commit 910163b

Please sign in to comment.