-
Notifications
You must be signed in to change notification settings - Fork 8
Launch Files
- Overview
- Launching Other Launch Files
roslaunch
Command-line Arguments- Building Command-line Interfaces (CLIs) for Python Nodes
- Passing Command-line Arguments to Python Nodes via
roslaunch
- Using the roslaunch Python API
As you'll know by now, we execute Nodes on a ROS network using the rosrun
command:
rosrun {package name} {script name}
Alternatively though, we also have the option of using roslaunch
:
roslaunch {package name} {launch file}
... where (as we learnt in Week 1) a {launch file}
is an xml
file with a .launch
file extension. Inside this we can ask ROS to do a number of different things from one single roslaunch
command-line call.
roslaunch
therefore offers a number of advantages over rosrun
:
- Multiple nodes can be executed simultaneously.
-
roslaunch
will launch the ROS Master for us, if isn't already running (so we don't have to manually callroscore
). - From within one
.launch
file, we can launch other.launch
files. - We can pass in additional arguments to launch certain things conditionally, or pass specific arguments to certain ROS nodes.
- As if that wasn't enough, there is also a
roslaunch
Python API, allowing us to launch nodes from within other (Python) nodes!
On this page we will talk about some features that may be useful for you.
Let's think back to the move_circle.py
node that we created in Week 2. We need a simulation of our robot active in order to run this, which we can enable with the following command (which you should be very familiar with by now):
$ roslaunch turtlebot3_gazebo turtlebot3_empty_world.launch
Suppose you wanted to save yourself some work and launch both the simulation and the move_circle.py
node at the same time...
-
Firstly, return to your
week2_navigation
package and create alaunch
directory (if you haven't done so previously):$ roscd week2_navigation $ mkdir launch $ cd launch/
-
Then create a launch file. You could call this whatever you want, but for the purposes of this example we'll create one called
circle.launch
:$ touch circle.launch
-
Inside this file add the following:
<launch> <include file="$(find turtlebot3_gazebo)/launch/turtlebot3_empty_world.launch" /> <node pkg="week2_navigation" type="move_circle.py" name="circle_node" output="screen" /> </launch>
The
<node>
tag should already be familiar to you, but the<include>
tag before it is new...This is what we use to launch other launch files. Here, we are locating the
turtlebot3_gazebo
package (usingfind
), and asking for theturtlebot3_empty_world.launch
file to be executed. -
From the command-line, execute your newly created launch file as follows:
$ roslaunch week2_navigation circle.launch
The TurtleBot3 Waffle will be launched in the "empty world" simulation, and the robot will start moving in a circle straight away!
A lot of the launch files that we have used throughout the lab course actually have Command-Line Arguments (CLAs) that can be (optionally) supplied when making the roslaunch
call. Consider the turtlebot3_empty_world.launch
file for instance...
Enter the full command and then press the SPACE
and TAB
keys on your keyboard like so:
roslaunch turtlebot3_gazebo turtlebot3_empty_world.launch[SPACE][TAB][TAB]
This will reveal the available command-line arguments for this launch file:
model x_pos y_pos z_pos
We can therefore optionally specify custom values for these parameters in order to change what happens when this launch file is executed. Try this, for example:
$ roslaunch turtlebot3_gazebo turtlebot3_empty_world.launch x_pos:=1 y_pos:=1
When the Gazebo simulation is launched, the robot will be located at coordinates (1, 1)
in the X-Y plane, rather than (0, 0)
, as would usually be the case.
Note: We assign values to
roslaunch
command-line arguments using the:=
operator.
This is all made possible through the use of <arg>
tags at the start of the launch file (just after the opening <launch>
tag):
<launch>
<arg name="x_pos" default="0.0"/>
<arg name="y_pos" default="0.0"/>
...
Within these tags we define two attributes:
-
name
: the name we want to give to the CLA (name="x_pos"
,name="y_pos"
etc...) -
default
: A default value in order to make the CLA optional (name="x_pos" default="0.0"
,name="y_pos" default="0.0"
, so by default, the robot will be located at coordinates(0, 0)
, unless we specify otherwise).
Suppose we have a very simple node, such as the publisher.py
node that we created in Week 1. This node publishes std_msgs/String
type messages to a topic called "chatter"
. Let's have a look at an alternative version of that node that takes in command-line arguments and publishes those to the "chatter"
topic instead.
Take a look at the publisher_cli.py
node from the com2009_examples
package. Make sure you also have this node installed locally by pulling down the most recent updates to the COM2009 repo:
$ cd ~/catkin_ws/src/COM2009 && git pull
Here, we use the Python argparse
module to allow us to work with arguments that are passed to the node from the command-line:
import argparse
We use this in the __init__()
method of the Publisher()
class to build a command-line interface (CLI) for the node:
cli = argparse.ArgumentParser(...)
cli.add_argument(...)
(See the code for more details)
Arguments that we define with a -
at the front will be optional, i.e. we don't have to provide these every time we run the node. We do, however, need to assign a default value for each optional argument in cases where no value is supplied, e.g.:
cli.add_argument("-colour", metavar="COL", type=String,
default="Blue",
...
The final step is to grab any arguments that are passed to this node when it is called. We use the rospy.myargv()
method here, so that this works regardless of whether we call the node using rosrun
or roslaunch
:
self.args = cli.parse_args(rospy.myargv()[1:])
Having defined the CLI above, argparse
then automatically generates help text for us! Try running the following command to see this in action:
$ rosrun com2009_examples publisher_cli.py -h
Note: You need to have a ROS Master running in order for this to work, so do this in another terminal instance by running the
roscore
command.
Run the node as it is (using rosrun
) and see what happens:
$ rosrun com2009_examples publisher_cli.py
Stop the node (using Ctrl+C
) and then run it again, but this time providing a value for the number
variable, via the CLI:
$ rosrun com2009_examples publisher_cli.py -number 1.5
Note: We can assign values to Python command-line arguments using a space (as above) or the
=
operator (-number=1.5
).
Stop the node again (using Ctrl+C
) and also stop the ROS Master that you enabled in another terminal instance.
Having built an CLI for our Python Node in the section above, we'll now look at how this all works when we call a node using roslaunch
instead.
-
First, create a launch file called
publisher_cli.launch
. It probably makes sense to create this one inside yourweek1_pubsub
package:$ roscd week1_pubsub/launch $ touch publisher_cli.launch
-
Inside this add the following:
<launch> <arg name="roslaunch_colour" default="Black" /> <arg name="roslaunch_number" default="0.999" /> <node pkg="com2009_examples" type="publisher_cli.py" name="publisher_cli_node" output="screen" args="-colour $(arg roslaunch_colour) -number $(arg roslaunch_number)" /> </launch>
Here, we are specifying command-line arguments for the launch file using
<arg>
tags, as discussed earlier:<arg name="roslaunch_colour" default="Black" /> <arg name="roslaunch_number" default="0.999" />
Next, we use a
<node>
tag to launch thepublisher_cli.py
node from thecom2009_examples
package. All of that should be familiar to you from Week 1. What's new here however is the additionalargs
attribute:args="-colour $(arg roslaunch_colour) -number $(arg roslaunch_number)"
This is how we pass the
roslaunch
command-line arguments to the CLI of our ROS Node:-
-colour
and-number
are the command-line arguments of the Python Node -
roslaunch_colour
androslaunch_number
are the command-line arguments of the launch file
-
-
Run the launch file without passing any arguments first, and see what happens:
$ roslaunch week1_pubsub publisher_cli.launch
-
Then try again, this time setting alternative values for the available command-line arguments:
$ roslaunch week1_pubsub publisher_cli.launch roslaunch_colour:=purple roslaunch_number:=4
The roslaunch API allows you to launch nodes (and other launch files) from inside your own nodes! The official documentation for the roslaunch API is the best place to go to find out how to use this, but here's a quick example, if you're short on time.
Recall the SLAM Exercise from Week 3. Here we had SLAM running in the background, while we drove our robot around a simulated environment (using the turtlebot3_teleop
node). Once we'd generated a full map we used a rosrun
command (in the terminal) to save a copy of this map:
$ rosrun map_server map_saver -f {map name}
Suppose we wanted to do this programmatically instead...
The following is an example of a very simple ROS node that does just that. Once the code below is executed it will launch the map_saver
node from the map_server
package in exactly the same way as above (but using the roslaunch API instead). The key difference here is that instead of {map name}
, we need to define the full file path for the map that we want to generate.
#!/usr/bin/env python3
import roslaunch
import rospy
map_path = "path/to/my_map"
rospy.init_node("map_getter", anonymous=True)
launch = roslaunch.scriptapi.ROSLaunch()
launch.start()
print(f"Saving map at time: {rospy.get_time()}...")
node = roslaunch.core.Node(package="map_server",
node_type="map_saver",
args=f"-f {map_path}")
process = launch.launch(node)
You could wrap this into one of your own ROS nodes, or even build a standalone node inside your package to call the map_saver
node when required. Perhaps you could even wrap the last three of the above commands inside a while loop and get the node to update a map file at regular intervals (say 5 seconds?), while your robot explores its environment...
COM2009/3009 Robotics Lab Course
Updated for the 2021-22 Academic Year
Dr Tom Howard | Multidisciplinary Engineering Education (MEE) | The University of Sheffield
The documentation within this Wiki is licensed under Creative Commons License CC BY-NC:
You are free to distribute, remix, adapt, and build upon this work (for non-commercial purposes only) as long as credit is given to the original author.