Pub Sub Example
Here we show how Sunset has been integrated onto physical hardware.
Last updated
Here we show how Sunset has been integrated onto physical hardware.
Last updated
For this section you might find it helpful to follow along with the code here.
Here is a video of the robot in particular.
Our robot is designed with four independently controlled mecanum wheels allowing for omni-directional motion. The system architecture employs a distributed approach where each motor controller operates as an individual node within the robot’s communication network. The topic / subscriber is integral to this setup, enabling systematic structure regarding how data flows through the system and how this is used to build high level behaviors.
In our network graph, these motor controllers are visualized as the "leaf" nodes, signifying their role as data endpoints in the larger robotic system. Each topic in the pub-sub system is treated as a discrete unit of computation, seamlessly passing messages and sensor readings along the chain of topics. This structure ensures that data flows logically from one topic to the next, mirroring the actual data pathways within the robot’s operational framework.
Mecanum Wheel Robotics System Network
By encapsulating each motor controller’s outputs into distinct topics, we maintain a clean and scalable system design, where extending functionality or troubleshooting becomes a matter of addressing specific units of computation within the pub-sub hierarchy.
Sunset is automatically able to determine that this node should have its execute method run first, then system time, then robot velocity, and finally once all the prerequisite data has been acquired, then can we compute the odometry update for this loop.
Topics in Sunset begin by inheriting from the Topic abstract class found in the architecture.architecture_relationships module.
Something important to note is that each instanciated topic must have a different name. For example if two roboclaw topics were to be spawned you must do something such as sensor1
sensor2
This is because sunset uses name as the ID used to mark things such as which message came from which Topic as well as which topic should go first in execution order. A runtime check is performed to help alleviate this type of error but due to the ease of bypassing this check it is recommended to pay special attention to topic names.
Here is the actual example for how a Topic can be created that periodically publishes some sensor value. In this case, each time the periodic method runs it obtains the estimated velocity from the motor controller and publishes them to its subscribers.
replace_message_with_log = True
Notice how the replace_message_with_log
field is true here. If you want to take advantage of sunsets logging / replay capabilities you must enable this for every root Topic only.
This tells sunset that this Topic is not a pure function and relies on state from API access that is not a receiving message. As such if we attempt to simulate this Topic there is no way to do that except for to use the logged information.
Your periodic method of Topics / Subscribers needs to be designed to execute quickly. You should also need to build it so that its designed to be called over and over again for the life time of your robot. To ensure safety, sunset does not use threads which means that you should not (and generally speaking cannot) implement your own event loops in the periodic methods.
Here is a rough pseudocode example of what is happening behind the scenes
See the scheduler
to see specific implementation details.
As a recap this is okay:
And this will break things:
The messaging bus of sunset allows us to distribute our computation into discrete, testable nodes which gives us confidence and understanding of our system. Here we will see how to utilize this system.
Here is an example of a Topic that computes the odometry position of the mecanum robot:
The important part here is what is occuring in subscriber_periodic
. Here the robot is reading the time constant from the schedulers timer and then getting the robot relative (x, y, omega)
velocities to compute with. Then the twist
is computed and the pose is transformed, creating the new pose. Once this has been done, the topic is now ready to be published by generate_message_periodic
which takes the pose field and extracts the necessary values and stores them as a message.
Notice how this topic relies on no other information other than what is passed into it. This means that it can be reconstructed purely based on the messages from other topics. For best usage with simulation, replace_message_with_log should be false (default value).
If you want simulation to work you must only use the SystemTime topic that is instantiated within the scheduler. See here for more information. We can see the system time topic's messages being used for dt
in the above example.
We create the message using floats instead of the SE3 pose object because floats are more easy to serialize to JSON for our log.