ROS2 nodes and interfaces
To run those example ensure the robot is not automatically performing some movement from lvl5. Select what levels to launch using the arguments. Example: ros2 launch motion_stack moonbot_zero.launch.py MS_up_to_level:=2
, this will launch levels 1 and 2.
Note
Ensure you sourced the workspace before running any of those commands: source ~Motion-Stack/install/setup.bash
Level 01: Joint
Important
This node’s Python code is meant to be specialized for your robot (through wrapping, overwriting, injecting …). Refer to Lvl1 specialization API to change the interface.
Is the glue between the motion stack and lower levels like Rviz, simulation or real robot. Its goal is to process joint states (sensor reading and motor commands). Handled joints are decided based on the URDF and/or launch parameters. It can be responsible for only one joint, one leg, one robot or all joints it receives.
- Source code:
Python:
motion_stack.core.lvl1_joint
Ros2 interface:
motion_stack.ros2.base_node.lvl1
Ros2 default node:
motion_stack.ros2.default_node.lvl1
- Topics:
joint_set
(Input from lvl2)JointState
: Goal state for the jointsjoint_read
(Output to lvl2)JointState
: Current state of the jointsjoint_commands
(Output to lvl0)JointState
: Command sent to the motorsjoint_states
(Input from lvl0)JointState
: Sensors reading of the joints
- Services:
advertise_joints
(Output)ReturnJointState
: Returns the names (in the URDF) of all joints being handled by that node. (Additionally returns the latest state data, with nan if no data is available. However, this should not be used as a replacement to joint_read.)
Caution
ROS2 Message JointState
does not guarantee the order, nor the existence of joints, nor the presence of each data array. Refer to the the doc.
Leveraging this, the motion stack uses a lazy communication strategy, hence a joint at rest will not be published at high frequency.
# List all joints handled by leg1 using:
ros2 service call /leg1/advertise_joints motion_stack_msgs/srv/ReturnJointState
# Read the angles:
ros2 topic echo /leg1/joint_read
# Send an angle of 1 rad:
ros2 topic pub /leg1/joint_set sensor_msgs/msg/JointState "{name: [joint1_2], position: [1.0], velocity: [], effort: []}"

Level 02: IK
This node loads the urdf to get all the kinematic information about its assigned leg. It computes the IK of the given target and outputs the joint states toward lvl1.
- Source code:
- Topics:
set_ik_target
(Input from lvl3)Transform
: Target command for the end effector of the leg. Relative to the body center (base_link
). (If less than 6 DoF leg, quaternion data is ignored.)tip_pos
(Output to lvl3)Transform
: Publishes the Transform of the leg’s end effector according to the joint angles reading.joint_set
(Output to lvl1)JointState
: see lvl1joint_read
(Input from lvl1)JointState
: see lvl1
# Send an IK target
ros2 topic pub /leg1/set_ik_target geometry_msgs/msg/Transform "{translation: {x: 400, y: 0, z: -100}, rotation: {x: 0.0, y: 0.0, z: 0.0, w: 1.0}}" -1
# Read the end effector position
ros2 topic echo /leg1/tip_pos

Level 03: Leg
This node handles long running trajectories, outputing IK targets. It does not hold any dimension information.
- Source code:
- Topics:
tip_pos
(Input from lvl2)Transform
: See lvl 02.set_ik_target
(Output to lvl2)Transform
: See lvl 02.
- Services:
rel_transl
(Input from lvl4)TFService
: Translates the tip of the leg linearly to the target. (Origin is base_link)shift
(Input from lvl4)TFService
: Translates the tip of the leg linearly to the target. (Origin is current tip position, origin orientation is similar to base_link)rel_hop
(Input from lvl4)TFService
: jumps the tip of the leg to the traget. Trajectory goes up, then moves above the target before going down onto the target. (Origin is base_link)rot
(Input from lvl4)TFService
: Rotates the leg tip linearly, BUT !!! around the center specified by the TF. (Origin is base_link)tip_pos
(Output to lvl4)ReturnVect3
: Returns the current position of the tip of the leg or the target if the tip is close to it. (Origin is base_link)
Note
Use shift
to rotate the leg tip with the center of rotation being the leg tip.
# send a straight shift motion 100 mm upward
ros2 service call /leg1/shift motion_stack_msgs/srv/TFService "{tf: {translation: {x: 0, y: 0, z: 100}, rotation: {x: 0.0, y: 0.0, z: 0.0, w: 1.0}}}"
# requests the tip position
ros2 service call /leg1/tip_pos motion_stack_msgs/srv/ReturnVect3
Level 04: Mover
Synchronizes several legs.
- Source code:
- Service:
body_tfshift
(Input from lvl5)TFService
: Translates the body by the given TF.get_targetset
(Input form lvl4s)ReturnTargetSet
: Returns the current target set of the robot (list of ee coordinates)legX/rel_transl
legX/shift
legX/rel_hop
legX/rot
(Output to lvl4): Refer to lvl4legX/tip_pos
(Input from lvl4)ReturnVect3
: Refer to lvl4.
cd ${ROS2_MOONBOT_WS}
. install/setup.bash
ros2 service call body_tfshift motion_stack_msgs/srv/TFService "{tf: {translation: {x: 0, y: 0, z: 100}, rotation: {x: 0.0, y: 0.0, z: 0.0, w: 1.0}}}"
ros2 service call body_tfshift motion_stack_msgs/srv/TFService "{tf: {translation: {x: 0, y: 0, z: -100}, rotation: {x: 0.0, y: 0.0, z: 0.0, w: 1.0}}}"
ros2 service call body_tfshift motion_stack_msgs/srv/TFService "{tf: {translation: {x: 0, y: 0, z: 100}, rotation: {x: 0.1, y: 0.0, z: 0.0, w: 1.0}}}"
ros2 service call body_tfshift motion_stack_msgs/srv/TFService "{tf: {translation: {x: 0, y: 0, z: 100}, rotation: {x: -0.1, y: 0.0, z: 0.0, w: 1.0}}}"
cd ${ROS2_MOONBOT_WS}
. install/setup.bash
ros2 service call /get_targetset motion_stack_msgs/srv/ReturnTargetSet