Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add ros2 node get_type_description initial implementation #846

Open
wants to merge 2 commits into
base: rolling
Choose a base branch
from

Conversation

emersonknapp
Copy link
Contributor

@emersonknapp emersonknapp commented Jul 28, 2023

Replaces #817

Implements a first pass at ros2 node get_type_description. It does its best to autofill the type hash so the user doesn't have to, but given the limitations of the current API this is only possible for msg on pub/sub topics, not for srv/action. For those, the hash must be provided manually by the user, though it is not discoverable via any existing API.

Sample output:

$ ros2 node get-type-description /talker rcl_interfaces/msg/Log
Waiting for service to become available...
Requesting type hash RIHS01_e28ce254ca8abc06abf92773b74602cdbf116ed34fbaf294fb9f81da9f318eac
Sending request...
Response successful:

rcl_interfaces/msg/Log
Fields:
  stamp: FIELD_TYPE_NESTED_TYPE (builtin_interfaces/msg/Time)
  level: FIELD_TYPE_UINT8
  name: FIELD_TYPE_STRING
  msg: FIELD_TYPE_STRING
  file: FIELD_TYPE_STRING
  function: FIELD_TYPE_STRING
  line: FIELD_TYPE_UINT32

Referenced Type Descriptions:

  builtin_interfaces/msg/Time
  Fields:
    sec: FIELD_TYPE_INT32
    nanosec: FIELD_TYPE_UINT32
$ ros2 node get-type-description /talker type_description_interfaces/srv/GetTypeDescription RIHS01_69b9c19c1021405984cc60dbbb1edceb147a6538b411d812ba6afabeed962cd5
Sending request...                                                                                                                                                                               
Response successful:

type_description_interfaces/srv/GetTypeDescription
Fields:
  request_message: FIELD_TYPE_NESTED_TYPE (type_description_interfaces/srv/GetTypeDescription_Request)
  response_message: FIELD_TYPE_NESTED_TYPE (type_description_interfaces/srv/GetTypeDescription_Response)
  event_message: FIELD_TYPE_NESTED_TYPE (type_description_interfaces/srv/GetTypeDescription_Event)

Referenced Type Descriptions:

  builtin_interfaces/msg/Time
  Fields:
    sec: FIELD_TYPE_INT32
    nanosec: FIELD_TYPE_UINT32

  service_msgs/msg/ServiceEventInfo
  Fields:
    event_type: FIELD_TYPE_UINT8
    stamp: FIELD_TYPE_NESTED_TYPE (builtin_interfaces/msg/Time)
    client_gid: FIELD_TYPE_UINT8_ARRAY - capacity 16
    sequence_number: FIELD_TYPE_INT64

  type_description_interfaces/msg/Field
  Fields:
    name: FIELD_TYPE_STRING
    type: FIELD_TYPE_NESTED_TYPE (type_description_interfaces/msg/FieldType)
    default_value: FIELD_TYPE_STRING

  type_description_interfaces/msg/FieldType
  Fields:
    type_id: FIELD_TYPE_UINT8 = 0
    capacity: FIELD_TYPE_UINT64
    string_capacity: FIELD_TYPE_UINT64
    nested_type_name: FIELD_TYPE_BOUNDED_STRING - string_capacity 255

  type_description_interfaces/msg/IndividualTypeDescription
  Fields:
    type_name: FIELD_TYPE_BOUNDED_STRING - string_capacity 255
    fields: FIELD_TYPE_NESTED_TYPE_UNBOUNDED_SEQUENCE (type_description_interfaces/msg/Field)

  type_description_interfaces/msg/KeyValue
  Fields:
    key: FIELD_TYPE_STRING
    value: FIELD_TYPE_STRING

  type_description_interfaces/msg/TypeDescription
  Fields:
    type_description: FIELD_TYPE_NESTED_TYPE (type_description_interfaces/msg/IndividualTypeDescription)
    referenced_type_descriptions: FIELD_TYPE_NESTED_TYPE_UNBOUNDED_SEQUENCE (type_description_interfaces/msg/IndividualTypeDescription)

  type_description_interfaces/msg/TypeSource
  Fields:
    type_name: FIELD_TYPE_STRING
    encoding: FIELD_TYPE_STRING
    raw_file_contents: FIELD_TYPE_STRING

  type_description_interfaces/srv/GetTypeDescription_Event
  Fields:
    info: FIELD_TYPE_NESTED_TYPE (service_msgs/msg/ServiceEventInfo)
    request: FIELD_TYPE_NESTED_TYPE_BOUNDED_SEQUENCE (type_description_interfaces/srv/GetTypeDescription_Request) - capacity 1
    response: FIELD_TYPE_NESTED_TYPE_BOUNDED_SEQUENCE (type_description_interfaces/srv/GetTypeDescription_Response) - capacity 1

  type_description_interfaces/srv/GetTypeDescription_Request
  Fields:
    type_name: FIELD_TYPE_STRING
    type_hash: FIELD_TYPE_STRING
    include_type_sources: FIELD_TYPE_BOOLEAN = True

  type_description_interfaces/srv/GetTypeDescription_Response
  Fields:
    successful: FIELD_TYPE_BOOLEAN
    failure_reason: FIELD_TYPE_STRING
    type_description: FIELD_TYPE_NESTED_TYPE (type_description_interfaces/msg/TypeDescription)
    type_sources: FIELD_TYPE_NESTED_TYPE_UNBOUNDED_SEQUENCE (type_description_interfaces/msg/TypeSource)
    extra_information: FIELD_TYPE_NESTED_TYPE_UNBOUNDED_SEQUENCE (type_description_interfaces/msg/KeyValue)

I think we should move forward with putting this in, as it covers the majority use case. Further discussion follows:

  1. It's awkward that we need to get names_and_types, then use that to go to get_x_info_by_topic in order to get TopicEndpointInfo to find the hash. Extend names_and_types to contain type hashes rmw#356 would make this part easier and support srv/action
  2. The fact that we need to do this graph-munging to "discover" the hash for a typename seems unnecessary. Given that a specific node has a 1:1 relationship of type name and type hash, it seems like GetTypeDescription should allow for either (but not both) to be empty in the request, so that the requesting client doesn't necessarily even need to know both pieces of information. From a user point of view, it seems very natural to say "you node, give me your description for mypackage/msg/Foo". It also seems like it could come up, though less commonly, to request "you node, what is the type that has hash RIHS01_e28ce254ca8abc06abf92773b74602cdbf116ed34fbaf294fb9f81da9f318eac". The current underlying implementation in rcl assumes the hash is the more important piece of information, using it as the key to its cache, but this usage reveals that it's probably more likely we wish to fetch this information by typename, using the hash as a potential validation. HOWEVER - all that aside, with Extend names_and_types to contain type hashes rmw#356 the "discovery" would be much simpler and so alleviate some of my concern with this point.

@emersonknapp
Copy link
Contributor Author

@clalancette could I get a reviewer assignment for this? (or @audrow @gbiggs friendly ping)

@emersonknapp emersonknapp force-pushed the emersonknapp/node-get-type-description branch from adba69e to 1703c0b Compare August 27, 2023 21:13
Signed-off-by: Emerson Knapp <[email protected]>
Copy link
Collaborator

@fujitatomoya fujitatomoya left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it would be nice to have test for topic.

if ns != 'msg':
raise RuntimeError(
f'Currently cannot auto-discover hashes for "{ns}", only "msg". '
'Please provide type_hash value to command.')
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

out of curiosity, how user can get type_hash for srv or action? that would be helpful to add here.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Unfortunately it is not exposed to user anywhere except in the source code ☹️ so you would already have the hash.

ros2/rmw#356 might help a little, but it's not the whole story. I think we need to change the signatures of Node::get_[service|topic]_names_and_types, or provide a new alternative API something like

struct InterfaceType {
std::string name;
rosidl_type_hash_t hash;
};

std::vector<std::string, std::vector<InterfaceType>> get_topic_names_and_types;


rclpy.init()
node = rclpy.create_node(f'{NODE_NAME_PREFIX}_td_requester_{os.getpid()}')
cli = node.create_client(GetTypeDescription, '/talker/get_type_description')
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
cli = node.create_client(GetTypeDescription, '/talker/get_type_description')
cli = node.create_client(GetTypeDescription, service_name)

Comment on lines +134 to +135
if not cli.wait_for_service(timeout_sec=5.0):
raise RuntimeError(f'Service {service_name} not found')
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

how about waiting until service is ready, unless user sends the signal to stop? i would have timeout option instead of static timeout that user cannot even see how long it waits.

Suggested change
if not cli.wait_for_service(timeout_sec=5.0):
raise RuntimeError(f'Service {service_name} not found')
cli.wait_for_service()

Copy link

@MichaelOrlov MichaelOrlov left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@emersonknapp In general looks good to me.
I only have a few minor concerns, especially about formatting output

formatting with - capacity 255 could be a bit confusing especially if the afterward will be printed out the default value. It is gonna look like FIELD_TYPE_INT8_ARRAY - capacity 25 = 0

I agree that if we had ros2/rmw#356 implemented it would significantly simplify the search for hashes.
I also thinking about why we require to provide a hash when doing a service request for get_type_description?
What if make the hash field optional and the service will return a list of types with hashes?
It will be a client-side responsibility to verify hashes and determine which type is really interesting.
IMO this way everybody's life would be much easier 😃

Update: @emersonknapp it would be nice to have some tests.

from type_description_interfaces.srv import GetTypeDescription


def print_field_type(ft):

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The function name print_field_type is a bit misleading since it doesn't print anything.
IMO it would be more appropriate to rename it to the get_field_type_str or something similar.

Comment on lines +42 to +45
if ft.string_capacity:
type_str += f' - string_capacity {ft.string_capacity}'
if ft.capacity:
type_str += f' - capacity {ft.capacity}'

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
if ft.string_capacity:
type_str += f' - string_capacity {ft.string_capacity}'
if ft.capacity:
type_str += f' - capacity {ft.capacity}'
if ft.string_capacity:
type_str += f' string_capacity({ft.string_capacity})'
if ft.capacity:
type_str += f' capacity({ft.capacity})'

IMO formatting with - capacity 255 could be a bit confusing especially if afterward will be printed out the dafult value. It is gonna look like FIELD_TYPE_INT8_ARRAY - capacity 25 = 0

Comment on lines +59 to +60
if field.default_value:
field_string += f' = {field.default_value}'

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
if field.default_value:
field_string += f' = {field.default_value}'
if field.default_value:
field_string += f' default({field.default_value})'

Comment on lines +67 to +70
print()
print('Referenced Type Descriptions:')
for rtd in td.referenced_type_descriptions:
print_individual_type_description(rtd, indent=2)

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Printing 'Referenced Type Descriptions:' make sense only if td.referenced_type_descriptions is not empty



def discover_hash_for_type(node, remote_node_name, type_name):
# TODO(emersonknapp) get hashes from names_and_types when that is implemented

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please add a reference to the tracking issue ros2/rmw#356 in the TODO comment.

@asancho99
Copy link

I need the hash to obtain the information of an interface in runtime using the GetTypeDescription service. With the type hash discovery for msg is really easy since just executing the "ros2 topic info -v /topic" I obtain the required hash to call the service. But afaik there is no type hash discovery for actions and services. Because of that, to use the GetTypeDescription I need to check the hash manually in the install folder, so it is only possible if you have access to the source code. Is there any plan to include the type hash discovery for actions and services to be able to take the advantage of knowing the structure of the interfaces in runtime (without access to the code) using the GetTypeDescription service?

@fujitatomoya
Copy link
Collaborator

Is there any plan to include the type hash discovery for actions and services to be able to take the advantage of knowing the structure of the interfaces in runtime (without access to the code) using the GetTypeDescription service?

AFAIK, nobody is currently working on this and neither the plan. ros2/ros2#1159 does not include this requirement, but there is the issue for that ros2/rmw#356

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants