-
Notifications
You must be signed in to change notification settings - Fork 1.8k
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
Message versioning and ROS 2 message translation #3465
base: main
Are you sure you want to change the base?
Conversation
a0c25d8
to
ac87603
Compare
ac87603
to
66ced5a
Compare
41873cf
to
825115e
Compare
3c869fb
to
2a3af80
Compare
2a3af80
to
22a480e
Compare
@hamishwillee I think the PR is in a review ready state. I left a few |
Co-authored-by: bkueng <beat-kueng@gmx.net>
22a480e
to
8d080d2
Compare
en/middleware/uorb.md
Outdated
Message versioning was introduced to address the challenges of maintaining compatibility across systems where different versions of message definitions may be in use, such as between PX4 and external systems like ROS 2 applications. | ||
|
||
This versioning mechanism supports the [ROS 2 Message Translation Node](../ros2/px4_ros2_msg_translation_node.md), which enables seamless communication between PX4 and ROS 2 applications; when different versions of message definitions are in use, the ROS 2 translation node ensures that messages can be converted and exchanged correctly. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is a little messy and perhaps can be simplified. In particular
- ROS2 Message Translation node supports versioning, not the versioning supporting the translation node.
Perhaps something like:
Message versioning was introduced to address the challenges of maintaining compatibility across systems where different versions of message definitions may be in use, such as between PX4 and external systems like ROS 2 applications. | |
This versioning mechanism supports the [ROS 2 Message Translation Node](../ros2/px4_ros2_msg_translation_node.md), which enables seamless communication between PX4 and ROS 2 applications; when different versions of message definitions are in use, the ROS 2 translation node ensures that messages can be converted and exchanged correctly. | |
Message versioning has been in introduced in PX4 v1.16 to make it easier to maintain compatibility between PX4 and ROS 2 versions. | |
This versioning mechanism uses the [ROS 2 Message Translation Node](../ros2/px4_ros2_msg_translation_node.md) to convert between different versions of messages, allowing seamless communication between PX4 and ROS 2 applications; when different versions of message definitions are in use. |
Perhaps add a diagram showing where the magic happens?
It would also be good "somewhere here" to mention any forward thinking about the ROS 2 native translation system that they have been talking about for a couple of years. Perhaps as a note at the end.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good idea, I've added a diagram and a note concerning ROS native translation.
ROS2 Message Translation node supports versioning, not the versioning supporting the translation node.
I may be thinking of this wrong, but in my mind message versioning works as a standalone feature (it's just about some messages having a version number incremented, others not). The translation node builds on top of versioned messages. In that sense, the translation supporting the versioning sounds backwards to me. How do you see this?
|
||
The steps include: | ||
|
||
- Copy the versioned message file (`.msg`/`.srv`) to `px4_msgs_old/msg/` (or `px4_msgs_old/srv/`) and add the current version to the file name. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Copy the versioned message file (
.msg
/.srv
)`
- what is .srv here?
- what is
px4_msgs_old
? How do they get that folder/repo? - "and add the" to "and append the current version information"
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
.srv
are the service definitions files (e.g. VehicleCommand.srv), as opposed to the.msg
message definition files. It is indeed confusing because thepx4_msgs
ROS2 package actually contains both the.msg
and.srv
definitions, and we use the term "message" loosely to designate specifically a message file, as well as either a message or service file. To clarify, I will change to:
- Copy the versioned message file (`.msg`/`.srv`) to `px4_msgs_old/msg/` (or `px4_msgs_old/srv/`) and add the current version to the file name. | |
- Copy the versioned `.msg` message file (or versioned `.srv` service file) to `px4_msgs_old/msg/` (or `px4_msgs_old/srv/`) and append the current version information to the file name. |
px4_msgs_old
is the ROS2 package that stores the versioned history of PX4 messages (and services), used by the translation node package. That folder is added to the PX4 repo in the final open PR, here. Currently it is empty, but in practice ifpx4_msgs
has a messageMyMessage.msg
which is atMESSAGE_VERSION = 3
, thenpx4_msgs_old
will contain two definitionsMyMessageV1.msg
andMyMessageV2.msg
. I will try to clarify
If a nested message definition changes, all messages including that message also require a version update. | ||
This is primarily important for services. | ||
::: | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What do I do if I have custom messages I want to version?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That would not be a problem, those custom messages would just need the MESSAGE_VERSION
field to be added and the file moved to the corresponding versioned/
folder. Updating those message would follow the same steps as updating vanilla px4 messages
|
||
- Copy the versioned message file (`.msg`/`.srv`) to `px4_msgs_old/msg/` (or `px4_msgs_old/srv/`) and add the current version to the file name. | ||
For example `msg/versioned/VehicleAttitude.msg` becomes `px4_msgs_old/msg/VehicleAttitudeV3.msg`. | ||
Update the existing translations that use the current topic version to the now old version. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Update the existing translations that use the current topic version to the now old version.
- What's a translation? Needs a definition.
- Am I replacing or adding a new one of these "whatever they are". I guess I'm asking if I make 3 versions of my message do I have three files? Or one that is updated three times?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good points, I've elaborated on this.
I guess I'm asking if I make 3 versions of my message do I have three files?
Exactly, 3 new versions of a message means adding three new translation files: each translation file bridging one version to the next
- Copy the versioned message file (`.msg`/`.srv`) to `px4_msgs_old/msg/` (or `px4_msgs_old/srv/`) and add the current version to the file name. | ||
For example `msg/versioned/VehicleAttitude.msg` becomes `px4_msgs_old/msg/VehicleAttitudeV3.msg`. | ||
Update the existing translations that use the current topic version to the now old version. | ||
For example `px4_msgs::msg::VehicleAttitude` becomes `px4_msgs_old::msg::VehicleAttitudeV3`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This example doesn't help answer my questions above - in what file would it be?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The existing translations would be under msg/translation_node/translations/*.h
. I've reworked the section to make it clearer
If a nested message definition changes, all messages including that message also require a version update. | ||
This is primarily important for services. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't know anything about nested message definitions. What are they?
Either way, perhaps a warning or tip?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nested messages are message definitions that include fields referencing other message types instead of using only the classic data types. For instance PositionSetpointTriplet.
Thanks I've added the example and made the paragraph into a warning
- Add a version translation by adding a new translation header. Examples: (TODO: update GitHub urls) | ||
- [`translations/example_translation_direct_v1.h`](https://github.com/PX4/PX4-Autopilot/blob/message_versioning_and_translation/msg/translation_node/translations/example_translation_direct_v1.h) | ||
- [`translations/example_translation_multi_v2.h`](https://github.com/PX4/PX4-Autopilot/blob/message_versioning_and_translation/msg/translation_node/translations/example_translation_multi_v2.h) | ||
- [`translations/example_translation_service_v1.h`](https://github.com/PX4/PX4-Autopilot/blob/message_versioning_and_translation/msg/translation_node/translations/example_translation_service_v1.h) | ||
- Include the added header in [`translations/all_translations.h`](https://github.com/PX4/PX4-Autopilot/blob/message_versioning_and_translation/msg/translation_node/translations/all_translations.h). | ||
|
||
For the second last step and for topics, there are two options: | ||
|
||
1. Direct translations: these translate a single topic between two different versions. | ||
This is the simpler case and should be preferred if possible. | ||
2. Generic case: this allows a translation between N input topics and M output topics. | ||
This can be used for merging or splitting a message. | ||
Or for example when moving a field from one message to another, a single translation should be added with the two older message versions as input and the two newer versions as output. | ||
This way there is no information lost when translating forward or backward. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I haven't looked at the example, but it seems to me that this might require some worked examples of each type. If the headers are specific to each type, perhaps move them under the following sections?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You're right, I've gone and added a worked example also to make the steps clearer
|
||
## Limitations | ||
|
||
- The current implementation depends on a service API that is not yet available in ROS Humble, and therefore does not support services when built for ROS Humble. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So what versions will it work for?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Jazzy or later
## Limitations | ||
|
||
- The current implementation depends on a service API that is not yet available in ROS Humble, and therefore does not support services when built for ROS Humble. | ||
- Services only support a linear history, i.e. no message splitting or merging |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Do you mean services as in something different than message/topics? Perhaps this is what you mean by .svr above? Is there information on what those mean anywhere?
Either way, so are you saying that services require a linear history, but you can split or merge messages as you like?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Correct, ROS distinguishes between services and topics like so: https://docs.ros.org/en/foxy/How-To-Guides/Topics-Services-Actions.html. In PX4, we have the srv/ services folder vs the msg/ topics folder.
Messages define the data format used to communicate over either topics or via services. The confusing part is that .srv
files define service messages, and .msg
files define topic messages (the file extension introduces confusion and make them seem more generic)
And finally yes, currently services require a linear history to translate while topics (.msg
) can be split and merged
Thanks for the comment, I realized I was also confused by this too. I'll do my best to make this clear in the docs
|
||
- The current implementation depends on a service API that is not yet available in ROS Humble, and therefore does not support services when built for ROS Humble. | ||
- Services only support a linear history, i.e. no message splitting or merging | ||
- Having both publishers and subscribers for two different versions of a topic is currently not handled by the translation node and would trigger infinite circular publications. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So a ROS application must compile against just one version of a topic? What if there are two ROS applications running on the network compiled against different versions - is this the problem case?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So a ROS application must compile against just one version of a topic?
Yes
What if there are two ROS applications running on the network compiled against different versions - is this the problem case?
No, only a specific subset, like:
- app 1: pub topic_v1, sub topic_v1
- app 2: pub topic_v2, sub topic_v2
In practice there's rarely (if any) setup that will run into this.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
OK, lots of comments here. They don't reflect that this is pretty good structure. Just I'm using myself as the straw man, and I don't know much ROS 2.
When you've had a look at the comments and tried address them, can you loop in Beat and Benja to have a look? I ask because I probably will be on holiday when you do this. Back around week of Jan 20th.
22a0459
to
1bf239e
Compare
1d84269
to
ab1c88d
Compare
Maybe worth adding to the 1.16 release wip page? @hamishwillee |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Nice work, thanks for going through this
|
||
## Limitations | ||
|
||
- The current implementation depends on a service API that is not yet available in ROS Humble, and therefore does not support services when built for ROS Humble. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Jazzy or later
|
||
- The current implementation depends on a service API that is not yet available in ROS Humble, and therefore does not support services when built for ROS Humble. | ||
- Services only support a linear history, i.e. no message splitting or merging | ||
- Having both publishers and subscribers for two different versions of a topic is currently not handled by the translation node and would trigger infinite circular publications. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So a ROS application must compile against just one version of a topic?
Yes
What if there are two ROS applications running on the network compiled against different versions - is this the problem case?
No, only a specific subset, like:
- app 1: pub topic_v1, sub topic_v1
- app 2: pub topic_v2, sub topic_v2
In practice there's rarely (if any) setup that will run into this.
No flaws found |
@hamishwillee Thanks for the thorough review! I've expanded on what was there, hopefully this addresses most of your comments. Let me know if parts are still unclear when you're back. In the meantime @beniaminopozzan would you like to have a look? Hamish told me might be interested in being kept in the loop |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks @GuillaumeLaine !
I just pointed out a few typo. Nothing else for me to report - just exited to see it finished!
Message versioning has been in introduced in PX4 v1.16 (main) to make it easier to maintain compatibility between PX4 and ROS 2 versions. | ||
|
||
Versioned messages include an additional field `uint32 MESSAGE_VERSION = x`, where `x` corresponds to the current version of the message. | ||
Versioned message files are stored separetly from their non-versioned counterpart. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Versioned message files are stored separetly from their non-versioned counterpart. | |
Versioned message files are stored separately from their non-versioned counterpart. |
For more on PX4 and ROS 2 communication, refer to the page about the [PX4-ROS 2 Bridge](../ros/ros2_comm.md). | ||
|
||
::: note | ||
ROS 2 plans to natively support message versioning in the future, but this is not implmemented yet. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ROS 2 plans to natively support message versioning in the future, but this is not implmemented yet. | |
ROS 2 plans to natively support message versioning in the future, but this is not implememented yet. |
|
||
## Overview | ||
|
||
The translation of messages from one definition version to another is possbile thanks to the introduction of [message versioning](../middleware/uorb.md#message-versioning). |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The translation of messages from one definition version to another is possbile thanks to the introduction of [message versioning](../middleware/uorb.md#message-versioning). | |
The translation of messages from one definition version to another is possible thanks to the introduction of [message versioning](../middleware/uorb.md#message-versioning). |
|
||
The translation of messages from one definition version to another is possbile thanks to the introduction of [message versioning](../middleware/uorb.md#message-versioning). | ||
|
||
The translation node has access to all message versions previously defined by PX4. It dynamically observes the DDS data space, monitoring the publications, subscriptions and services originating from either PX4 via the [uXRCE-DDS Bridge](../middleware/uxrce_dds.md), or ROS2 applications. When necessary, it converts messages to the current versions expected by both applications and PX4, ensuring compatibility. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The translation node has access to all message versions previously defined by PX4. It dynamically observes the DDS data space, monitoring the publications, subscriptions and services originating from either PX4 via the [uXRCE-DDS Bridge](../middleware/uxrce_dds.md), or ROS2 applications. When necessary, it converts messages to the current versions expected by both applications and PX4, ensuring compatibility. | |
The translation node has access to all message versions previously defined by PX4. It dynamically observes the DDS data space, monitoring the publications, subscriptions and services originating from either PX4 via the [uXRCE-DDS Bridge](../middleware/uxrce_dds.md), or ROS 2 applications. When necessary, it converts messages to the current versions expected by both applications and PX4, ensuring compatibility. |
from px4_msgs.msg import VehicleCommand, VehicleAttitude | ||
|
||
# Helper function to get the message version suffix | ||
# The correct message version is directly inferred from the message defintion |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
# The correct message version is directly inferred from the message defintion | |
# The correct message version is directly inferred from the message definition |
A __versioned message__ is a message for which changes are tracked and each change results in a version bump, with the previous state of the definition being stored in history. | ||
The latest version of every message is stored in `msg/versioned/` for topics (or `srv/versioned` for services), and all older versions are stored in `msg/px4_msgs_old/msg/` (or `msg/px4_msgs_old/srv/`). | ||
|
||
A __version translation__ defines a bidrectional mapping of the contents of one or more message definition across different versions. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A __version translation__ defines a bidrectional mapping of the contents of one or more message definition across different versions. | |
A __version translation__ defines a bidirectional mapping of the contents of one or more message definition across different versions. |
- Each file includes a `MESSAGE_VERSION` field. | ||
- File names reflect the message's version with a suffix (e.g., `V1`, `V2`). | ||
|
||
Example directory structure (coherant with example above): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Example directory structure (coherant with example above): | |
Example directory structure (coherent with example above): |
- Each translation (direct or generic) is a single `.h` header file. | ||
- The header `all_translation.h` acts as the main header, and includes all subsequent translation headers. | ||
|
||
Example directory structure (coherant with example above): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Example directory structure (coherant with example above): | |
Example directory structure (coherent with example above): |
This documents the newly versioned subset of PX4 messages, and the ROS 2 translation node that enables seamless communication between PX4 and ROS 2 applications that may be using different message definition versions.
Related Work
To Do
TODO
s in the markdown of this PR