tl;dr I’m looking for feedback from ROS 2 users on this new for-loop action for launch (XML/YAML/Python), which can repeat entities N times based on the value of a launch argument, and provides an index that can be used for substitutions: Add ForLoop action to repeat entities with an index by christophebedard · Pull Request #802 · ros2/launch · GitHub. Please post comments about the implementation on the PR and general comments about the solution here.
Being able to repeat entities (e.g., nodes) N times based on the value of a launch argument is a pretty commonly-requested feature. Example: ros2/launch#499 and [poll] Interest in ros2 launch action to support for-loops (e.g. IncludeNLaunchDescriptions())?. Some current solutions/workarounds include:
OpaqueFunction
: Launch Loop Support? [Feature Request] · Issue #499 · ros2/launch · GitHub- Recursion: GitHub - MetroRobots/rosetta_launch: A guide to understanding launch files in ROS 1 and ROS 2
While these work, they involve some boilerplate code that has to be repeated in each launch file, and they make launch files more complex.
I’m proposing a new ForLoop
action (ros2/launch#802), which also supports substituting the index value, e.g., $(index i)
in XML or YAML. Examples:
XML
<launch>
<arg name="num_robots" default="2" />
<for len="$(var num_robots)" name="i" >
<node pkg="test_tracetools" exec="test_ping" namespace="/pingpong_$(index i)" output="screen" />
<node pkg="test_tracetools" exec="test_pong" namespace="/pingpong_$(index i)" output="screen" />
</for>
</launch>
YAML
launch:
- arg:
name: num_robots
default: '2'
- for:
len: $(var num_robots)
name: i
children:
- node:
pkg: test_tracetools
exec: test_ping
namespace: /pingpong_$(index i)
output: screen
- node:
pkg: test_tracetools
exec: test_pong
namespace: /pingpong_$(index i)
output: screen
Python
import launch
import launch_ros
def for_i(i: int):
return [
launch_ros.actions.Node(
package='test_tracetools',
executable='test_ping',
output='screen',
namespace=['/pingpong_', str(i)],
),
launch_ros.actions.Node(
package='test_tracetools',
executable='test_pong',
output='screen',
namespace=['/pingpong_', str(i)],
),
]
def generate_launch_description():
return launch.LaunchDescription([
launch.actions.DeclareLaunchArgument('num_robots', default_value='2'),
launch.actions.ForLoop(launch.substitutions.LaunchConfiguration('num_robots'), function=for_i),
])
We discussed this a few months ago in a PMC meeting and I already got some feedback from some users, but I thought I’d post here to get more feedback from other ROS 2 launch users.