- Introduction
- Dependency installation, source build and setup
- Simulation examples
- Troubleshooting
- TODOs
Free fleet is a python implementation of the Open-RMF Fleet Adapter, based on the fleet_adapter_template
. It uses zenoh
as a communication layer between each robot and the fleet adapter, allowing access and control over the navigation stacks of the robots.
Using zenoh
bridges to pipe the necessary ROS 2 / 1 messages between each robot and the free_fleet_adapter
, users have the flexibility to configure and customize their network setups accordingly following the official guide. Examples provided in this repository are using these configurations, do take note of the selective topics that are required for the free_fleet_adapter
to work. The zenoh
configuration conveniently allows users to filter and limit the rate of messages based on topics as well, which will be helpful in deployments with limited network bandwidth.
Each robot's navigation stack is expected to be non-namespaced, while its zenoh
bridge is expected to be set up with it's robot name as the namespace. This allows the free_fleet_adapter
to integrate with each robot in the fleet individually using zenoh
namespaces.
Supports
- Ubuntu 24.04
- ROS 2 Jazzy
- rmw-cyclonedds-cpp
- Open-RMF on main
- zenoh-bridge-ros2dds v1.1.0
- zenoh-bridge-ros1 main
- zenoh router
We recommend setting up zenoh-bridge-ros2dds
with the released standalone binaries. After downloading the appropriate released version and platform, extract and use the standalone binaries as is. For source builds of zenoh-bridge-ros2dds
, please follow the official guides.
As for zenoh-bridge-ros1
, a custom fork is currently used to support bridge namespaces, and requires to be built from source. Once the changes have been merged upstream, this will be updated.
Most of the tests have been performed using rmw-cyclonedds-cpp
, while other RMW implementations have shown varying results. Support and testing with other RMW implementations will be set up down the road.
System dependencies,
sudo apt update && sudo apt install python3-pip ros-jazzy-rmw-cyclonedds-cpp
The dependencies eclipse-zenoh
, pycdr2
, rosbags
are available through pip
. Users can choose to set up a virtual environment, or --break-system-packages
by performing the installation directly.
pip3 install pip install eclipse-zenoh==1.1.0 pycdr2 rosbags --break-system-packages
Install zenohd
from the official guide.
Note
If an Open-RMF workspace has already been set up, users can choose to only set up an overlay workspace, which reduces build time. The following steps will assume a fresh new workspace is required.
Set up workspace, install dependencies and build,
mkdir -p ~/ff_ws/src
wget https://raw.githubusercontent.com/open-rmf/rmf/main/rmf.repos
vcs import ~/ff_ws/src < rmf.repos
cd ~/ff_ws/src
git clone https://github.com/open-rmf/free_fleet
# Install dependencies
cd ~/ff_ws
rosdep install --from-paths src --ignore-src --rosdistro $ROS_DISTRO -yr
# Build
colcon build --cmake-args -DCMAKE_BUILD_TYPE=Release
Download and extract standalone binaries for zenoh-bridge-ros2dds
(optionally zenohd
if a non-latest version is desired) with the correct architecture, system setup and version. The following example instructions are for x86_64-unknown-linux-gnu
,
# Change preferred zenoh version here
export ZENOH_VERSION=1.1.0
# Download and extract zenoh-bridge-ros2dds release
wget -O zenoh-plugin-ros2dds.zip https://github.com/eclipse-zenoh/zenoh-plugin-ros2dds/releases/download/$ZENOH_VERSION/zenoh-plugin-ros2dds-$ZENOH_VERSION-x86_64-unknown-linux-gnu-standalone.zip
unzip zenoh-plugin-ros2dds.zip
# If using released standalone binaries of zenoh router, download and extract the release
# wget -O zenoh.zip https://github.com/eclipse-zenoh/zenoh/releases/download/$ZENOH_VERSION/zenoh-$ZENOH_VERSION-x86_64-unknown-linux-gnu-standalone.zip
# unzip zenoh.zip
Examples for running a single robot or multiple robots in simulation has been up in free_fleet_examples
, along with example configuration files for zenoh
as well as fleet configuration files for free_fleet_adapter
.
For ROS 2, simulations will be launched using the nav2_bringup
package. Since the turtlebot3_gazebo
package is not being released past jazzy, users will need to clone the package to access the gazebo models,
sudo apt update && sudo apt install ros-jazzy-nav2-bringup
git clone https://github.com/ROBOTIS-GIT/turtlebot3_simulations ~/turtlebot3_simulations
This simulates running an isolated (by ROS_DOMAIN_ID
) turtlebot3 with a ROS 2 navigation stack, and setting up RMF with free_fleet_adapter
(on a different ROS_DOMAIN_ID
), allowing the fleet adapter to command the robot via a configured zenoh-bridge-ros2dds
with the namespace nav2_tb3
.
Launch simulation and set up the initial position of the robot (see gif),
source /opt/ros/jazzy/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
export GAZEBO_MODEL_PATH=$GAZEBO_MODEL_PATH:~/turtlebot3_simulations/turtlebot3_gazebo/models
# Launch the simulation
ros2 launch nav2_bringup tb3_simulation_launch.py headless:=0
# Or launch headless
# ros2 launch nav2_bringup tb3_simulation_launch.py
Start zenoh
router,
zenohd
# If using released standalaone binaries
# cd PATH_TO_EXTRACTED_ZENOH_ROUTER
# ./zenohd
Start zenoh-bridge-ros2dds
with the appropriate zenoh client configuration,
source /opt/ros/jazzy/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
cd PATH_TO_EXTRACTED_ZENOH_BRIDGE
./zenoh-bridge-ros2dds -c ~/ff_ws/src/free_fleet/free_fleet_examples/config/zenoh/nav2_tb3_zenoh_bridge_ros2dds_client_config.json5
Listen to transforms over zenoh
,
source ~/ff_ws/install/setup.bash
ros2 run free_fleet_examples nav2_get_tf.py \
--namespace nav2_tb3
Start a navigate_to_pose
action over zenoh
, using example values,
source ~/ff_ws/install/setup.bash
ros2 run free_fleet_examples nav2_send_navigate_to_pose.py \
--frame-id map \
--namespace nav2_tb3 \
-x 1.808 \
-y 0.503
Start the RMF core packages on a different ROS_DOMAIN_ID
to simulate running on a different machine,
source ~/ff_ws/install/setup.bash
export ROS_DOMAIN_ID=55
ros2 launch free_fleet_examples turtlebot3_world_rmf_common.launch.xml
Launch the free_fleet_adapter
with the current example's configurations, verify that nav2_tb3
has been added to fleet turtletbot3
.
source ~/ff_ws/install/setup.bash
export ROS_DOMAIN_ID=55
ros2 launch free_fleet_examples nav2_tb3_simulation_fleet_adapter.launch.xml
# Or launch with the rmf-web API server address
# ros2 launch free_fleet_examples nav2_tb3_simulation_fleet_adapter.launch.xml server_uri:="ws://localhost:8000/_internal"
Dispatch an example RMF patrol tasks using rmf-web
on the same ROS_DOMAIN_ID
as the RMF core packages, or use the dispatch_patrol
script,
source ~/ff_ws/install/setup.bash
export ROS_DOMAIN_ID=55
ros2 run rmf_demos_tasks dispatch_patrol \
-p north_west north_east south_east south_west \
-n 2 \
-st 0
Note
This multi-robot simulation example is only for testing purposes, as it is a different setup than free_fleet
is intended to be used. The simulation spawns 2 already namespaced robots, while the free_fleet
architecture expects individual non-namespaced robots to be partnered with a namespaced zenoh-bridge-ros2dds
.
In this example, there will only be one non-namespaced zenoh bridge for both robots, which will produce the same zenoh message outputs as 2 individual namespaced zenoh bridge with non-namespaced robots. This allows the free_fleet_adapter
to work with both robots on the same simulation.
Launch simulation, start the robots, and set up the initial positions (see gif),
source /opt/ros/jazzy/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
export GAZEBO_MODEL_PATH=$GAZEBO_MODEL_PATH:~/turtlebot3_simulations/turtlebot3_gazebo/models
ros2 launch nav2_bringup unique_multi_tb3_simulation_launch.py
Start zenoh
router,
zenohd
# If using released standalaone binaries
# cd PATH_TO_EXTRACTED_ZENOH_ROUTER
# ./zenohd
Start zenoh-bridge-ros2dds
with the appropriate zenoh client configuration,
source /opt/ros/jazzy/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
cd PATH_TO_EXTRACTED_ZENOH_BRIDGE
./zenoh-bridge-ros2dds -c ~/ff_ws/src/free_fleet/free_fleet_examples/config/zenoh/nav2_unique_multi_tb3_zenoh_bridge_ros2dds_client_config.json5
Start the RMF core packages on a different ROS_DOMAIN_ID
to simulate running on a different machine,
source ~/ff_ws/install/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
export ROS_DOMAIN_ID=55
ros2 launch free_fleet_examples turtlebot3_world_rmf_common.launch.xml
Launch the free_fleet_adapter
with the current example's configurations, verify that nav2_tb3
has been added to fleet turtlebot3
.
source ~/ff_ws/install/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
export ROS_DOMAIN_ID=55
ros2 launch free_fleet_examples nav2_unique_multi_tb3_simulation_fleet_adapter.launch.xml
# Or launch with the rmf-web API server address
# ros2 launch free_fleet_examples nav2_unique_multi_tb3_simulation_fleet_adapter.launch.xml server_uri:="ws://localhost:8000/_internal"
Dispatch example RMF patrol tasks using rmf-web
on the same ROS_DOMAIN_ID
as the RMF core packages, or use the dispatch_patrol
scripts, which will cause the robot to negotiate as they perform their tasks.
source ~/ff_ws/install/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
export ROS_DOMAIN_ID=55
# robot1 to run clockwise around the map
ros2 run rmf_demos_tasks dispatch_patrol \
-p north_west north_east south_east south_west \
-n 3 \
-st 0 \
-F turtlebot3 \
-R robot1
# robot2 to run anti-clockwise around the map
ros2 run rmf_demos_tasks dispatch_patrol \
-p south_west south_east north_east north_west \
-n 3 \
-st 0 \
-F turtlebot3 \
-R robot2
Warning
The Nav1 integration has only been tested in simulation and in ROS 1 Noetic, and is currently still using a fork of zenoh-plugin-ros1, to support bridge namespacing. This will be updated after contributions to upstream has been made.
Check out the docker compose integration tests for an overview of how the integration can be set up.
On the machine where the free fleet adapter will run, start a zenoh
router,
zenohd
# If using released standalaone binaries
# cd PATH_TO_EXTRACTED_ZENOH_ROUTER
# ./zenohd
In the ROS 1 Noetic environment, set up all the prerequisites as mentioned in the official guide, and start the turtlebot3 simulation. See the relevant docker file for reference.
source /opt/ros/noetic/setup.bash
export TURTLEBOT3_MODEL=burger
roslaunch turtlebot3_gazebo turtlebot3_world.launch
In the ROS 1 Noetic environment, bringup the Nav1 stack with the prepared map in navigation2. See the relevant docker file for reference.
# prepare the map
git clone https://github.com/ros-navigation/navigation2
source /opt/ros/noetic/setup.bash
export TURTLEBOT3_MODEL=burger
roslaunch turtlebot3_navigation turtlebot3_navigation.launch map_file:=/PATH_TO_navigation2/nav2_bringup/maps/tb3_sandbox.yaml
In the ROS 1 Noetic environment, set up prerequisites of zenoh-plugin-ros1, build zenoh-bridge-ros1
in release, and start it with the provided config in examples. See the relevant docker file for reference.
# Get the config file
git clone https://github.com/open-rmf/free_fleet
# Build the bridge
git clone --recursive https://github.com/aaronchongth/zenoh-plugin-ros1
cd zenoh-plugin-ros1
cargo build --package zenoh-bridge-ros1 --bin zenoh-bridge-ros1 --release
# Use cargo run, or just run the executable directly
source /opt/ros/noetic/setup.bash
./target/release/zenoh-bridge-ros1 -c PATH_TO_free_fleet/free_fleet_examples/config/zenoh/nav1_tb3_zenoh_bridge_ros1_client_config.json5
On the machine where the free fleet adapter will run, start the common launch files and the free fleet adapter,
source ~/ff_ws/install/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
ros2 launch free_fleet_examples turtlebot3_world_rmf_common.launch.xml
Launch the free_fleet_adapter
with the current example's configurations, verify that nav1_tb3
has been added to fleet turtlebot3
.
source ~/ff_ws/install/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
ros2 launch free_fleet_examples nav1_tb3_simulation_fleet_adapter.launch.xml
# Or launch with the rmf-web API server address
# ros2 launch free_fleet_examples nav1_tb3_simulation_fleet_adapter.launch.xml server_uri:="ws://localhost:8000/_internal"
Dispatch example RMF patrol tasks using rmf-web
on the same ROS_DOMAIN_ID
as the RMF core packages, or use the dispatch_patrol
scripts, which will cause the robot to negotiate as they perform their tasks.
source ~/ff_ws/install/setup.bash
export RMW_IMPLEMENTATION=rmw_cyclonedds_cpp
export ROS_DOMAIN_ID=55
# nav1_tb3 to run clockwise around the map
ros2 run rmf_demos_tasks dispatch_patrol \
-p north_west north_east south_east south_west \
-n 3 \
-st 0 \
-F turtlebot3 \
-R nav1_tb3
-
Looking for the legacy implementation of
free_fleet
? Check out the tag 1.3.0, or thelegacy
branch. -
free_fleet_adapter
can't seem to control the robots? Check if the zenoh messages are going through using the testing scripts infree_fleet_examples
. For ROS 2 navigation stacks, make sure that thezenoh-bridge-ros2dds
is launched with the sameRMW_IMPLEMENTATION
andROS_DOMAIN_ID
as the robot's navigation stack, otherwise no messages will be passed through the bridge. -
Failing to start
free_fleet_adapter
due to missing API inrmf_fleet_adapter_python
? This may be due to using outdatedrmf_fleet_adapter_python
released binaries, either perform asudo apt update && sudo apt upgrade
, or build RMF from source following the official guide. -
Simulations don't seem to work properly anymore? Try
ros2 deamon stop
,ros2 daemon start
, or explicitly kill theros
andgazebo
processes, or restart your machine. It's been noticed that if the ROS 2 or gazebo process are not terminated properly (happens rarely), the network traffic between the simulation robots and the fleet adapter get affected. -
ROS 1 Navigation stack simulation does not seem to work as expected? Check out the integration tests docker compose, as well as the simulation and bringup docker files, for any missing dependencies.
-
Why does RMF not run with
use_sim_time:=true
in the examples? This is because it is on a differentROS_DOMAIN_ID
than the simulation, therefore it will not have access to the simulationclock
topic, the examples running RMF,free_fleet_adapter
and the tasks will not be using sim time. -
For potential bandwidth issues, especially during multirobot sim example, spinning up a dedicated zenoh router and routing the
zenoh-bridge-ros2dds
manually to it, could help alleviate such issues. -
If
zenoh
messages are not received, make sure the versions between theeclipse-zenoh
inpip
,zenoh-bridge-ros2dds
andzenohd
are all the same. If the debian binary releases ofzenohd
have breaking changes, and the repo has not yet migrate to the newer version, please open an issue ticket and we will look into migrating as soon as possible. In the meantime, using an older standalone release ofzenohd
would be a temporary workaround. Our integration tests will attempt to catch these breaking changes too. -
zenohd
address already in use. This is most likely due to therest-http-port
which uses port 8000 by default, and might cause a conflict with other systems, for examplermf-web
's API server. Runzenohd --rest-http-port 8001
to change it to 8001 or anything else.
- attempt to optimize tf messages (not all are needed)
- custom actions to be abstracted
- map switching support
- end-to-end testing with Open-RMF
- test replanning behavior
- support for Rolling
- docker images
- releases
- testing and support for other RMW implementations