Launch Gazebo from ROS 2#
Gazebo can be launched from a ROS 2 launch system in multiple ways:
Using the launch files included in#
The package ros_gz_sim contains two launch files named gz_server.launch.py
and gz_sim.launch.py. You can use them to start Gazebo server or Gazebo (server and GUI)
respectively.
ros2 launch ros_gz_sim gz_sim.launch.py gz_args:=empty.sdf
Or you can just start the server:
ros2 launch ros_gz_sim gz_server.launch.py world_sdf_file:=empty.sdf
Consult the argument block of each launch file here and here to learn about the different parameters that are accepted by each launch file.
Using a custom launch file.#
XML#
It’s also possible to start Gazebo from your custom launch file. For that
purpose we have created the custom <gz_server/> tag that can be used from your
XML launch file. In this case, the arguments are passed as attributes
within this tag. Here’s an example for launching Gazebo server:
<launch>
  <arg name="world_sdf_file" default="empty.sdf" />
  <arg name="world_sdf_string" default="" />
  <arg name="container_name" default="ros_gz_container" />
  <arg name="create_own_container" default="False" />
  <arg name="use_composition" default="False" />
  <gz_server
    world_sdf_file="$(var world_sdf_file)"
    world_sdf_string="$(var world_sdf_string)"
    container_name="$(var container_name)"
    create_own_container="$(var create_own_container)"
    use_composition="$(var use_composition)">
  </gz_server>
</launch>
In this case the <gz_server> parameters are read from the command line. That’s
an option but not strictly necessary as you could decide to hardcode some of the
values or not even use all the parameters.
Python#
Python launch files provide more low-level customization and logic compared to XML launch files. For example, you can set environment variables and include Python specific functions and logic. In the following example, the user can replace the example package, world, and bridged topic with their own. This is intended as a scaffolding more than something that can be run on its own.
from launch import LaunchDescription
from launch_ros.actions import Node
from launch.actions import SetEnvironmentVariable, IncludeLaunchDescription
from launch.launch_description_sources import PythonLaunchDescriptionSource
from launch.substitutions import PathJoinSubstitution
from launch_ros.substitutions import FindPackageShare
def generate_launch_description():
    ros_gz_sim_pkg_path = get_package_share_directory('ros_gz_sim')
    example_pkg_path = FindPackageShare('example_package')  # Replace with your own package name
    gz_launch_path = PathJoinSubstitution([pkg_ros_gz_sim, 'launch', 'gz_sim.launch.py'])
    return LaunchDescription([
        SetEnvironmentVariable(
            'GZ_SIM_RESOURCE_PATH',
            PathJoinSubstitution([example_pkg_path, 'models'])
        ),
        SetEnvironmentVariable(
            'GZ_SIM_PLUGIN_PATH',
            PathJoinSubstitution([example_pkg_path, 'plugins'])
        ),
        IncludeLaunchDescription(
            PythonLaunchDescriptionSource(gz_launch_path),
            launch_arguments={
                'gz_args': [PathJoinSubstitution([example_pkg_path, 'worlds/example_world.sdf'])],  # Replace with your own world file
                'on_exit_shutdown': 'True'
            }.items(),
        ),
        # Bridging and remapping Gazebo topics to ROS 2 (replace with your own topics)
        Node(
            package='ros_gz_bridge',
            executable='parameter_bridge',
            arguments=['/example_imu_topic@sensor_msgs/msg/Imu@gz.msgs.IMU',],
            remappings=[('/example_imu_topic',
                         '/remapped_imu_topic'),],
            output='screen'
        ),
    ])
Here’s another example using a higher level action from Python to launch gzserver:
from launch import LaunchDescription
from launch.actions import DeclareLaunchArgument
from launch.substitutions import LaunchConfiguration
from ros_gz_sim.actions import GzServer
def generate_launch_description():
    declare_world_sdf_file_cmd = DeclareLaunchArgument(
        'world_sdf_file', default_value='',
        description='Path to the SDF world file')
    # Create the launch description and populate
    ld = LaunchDescription([
        GzServer(
            world_sdf_file=LaunchConfiguration('world_sdf_file')
        ),
    ])
    # Declare the launch options
    ld.add_action(declare_world_sdf_file_cmd)
    return ld
Launching with ros_gz_bridge#
An example launch file for XML can be viewed here An example launch file for Python can be viewed here
Example command for directly using these launch files from the terminal:
ros2 launch ros_gz_sim ros_gz_sim.launch.py world_sdf_file:=empty.sdf bridge_name:=ros_gz_bridge config_file:=<path_to_your_YAML_file> use_composition:=True create_own_container:=True
In the above launch files you may notice that the create_own_container argument for ros_gz_bridge is hardcoded to False. This has been done to prevent two duplicate containers from getting created (one for gz_server and another one for ros_gz_bridge), and instead make ros_gz_bridge use the container created by gz_server. More info about this can be viewed here
More info about ros_gz_bridge can be viewed here.
More info about composition can be viewed here.
Further Considerations for ROS2 Control#
If you’re planning to use ros2_control with Gazebo, please take a look at the example launch files in the gz_ros2_control repository
It is essential to publish the /clock topic for the controller_manager to function correctly:
 gz_bridge = Node(
     package="ros_gz_bridge",
     executable="parameter_bridge",
     arguments=['/clock@rosgraph_msgs/msg/Clock[gz.msgs.Clock'],
     parameters=[{
         "qos_overrides./tf_static.publisher.durability": "transient_local"
     }],
     output="screen",
 )
If you do not publish the /clock topic, the controller_manager will issue warnings or errors such as:
   [gazebo-1] [WARN] [1744219953.983130822] [controller_manager]: No clock received, using time argument instead! Check your node's clock configuration (use_sim_time parameter) and if a valid clock source is available.
Timing Issues#
By default, the controller_manager launched by gz_ros2_control has use_sim_time=true. If for any reason this is set to false, it will fall back to the system clock.
This results in logs like:
 [gazebo-1] [INFO] [1744209678.974210234] [gz_ros_control]: Loading controller_manager
 [gazebo-1] [INFO] [1744209679.000651931] [controller_manager]: Using Steady (Monotonic) clock for triggering controller manager cycles.
Eventually leading to a fatal error:
 [gazebo-1] terminate called after throwing an instance of 'std::runtime_error'
 [gazebo-1]   what():  can't compare times with different time sources
Ensure use_sim_time is correctly set to true when working with simulation time to avoid such mismatches.