About parameters in ROS 2
Table of Contents
Overview
Parameters in ROS are associated with individual nodes. Parameters are used to configure nodes at startup (and during runtime), without changing the code. The lifetime of a parameter is tied to the lifetime of the node (though the node could implement some sort of persistence to reload values after restart).
Parameters are addressed by node name, node namespace, parameter name, and parameter namespace. Providing a parameter namespace is optional.
Each parameter consists of a key, a value, and a descriptor. The key is a string and the value is one of the following types: bool, int64, float64, string, byte[], bool[], int64[], float64[] or string[]. By default all descriptors are empty, but can contain parameter descriptions, value ranges, type information, and additional constraints.
For an hands-on tutorial with ROS parameters see Understanding ROS 2 parameters.
Parameters background
Declaring parameters
By default, a node needs to declare all of the parameters that it will accept during its lifetime. This is so that the type and the name of the parameter are well-defined at node startup time, which reduces the chances of misconfiguration later on. See Using parameters in a class (C++) or Using parameters in a class (Python) for tutorials on declaring and using parameters from a node.
For some types of nodes, not all of the parameters will be known ahead of time.
In these cases, the node can be instantiated with allow_undeclared_parameters
set to true
, which will allow parameters to be get and set on the node even if they haven’t been declared.
Parameter types
Each parameter on a ROS 2 node has one of the pre-defined parameter types as mentioned in the Overview. By default, attempts to change the type of a declared parameter at runtime will fail. This prevents common mistakes, such as putting a boolean value into an integer parameter.
If a parameter needs to be multiple different types, and the code using the parameter can handle it, this default behavior can be changed.
When the parameter is declared, it should be declared using a ParameterDescriptor
with the dynamic_typing
member variable set to true
.
Parameter callbacks
A ROS 2 node can register two different types of callbacks to be informed when changes are happening to parameters. The reason that there are two types of callbacks is to have a chance to intervene before the parameter change happens, and to have a chance to react after the parameter change happens. A node can register for both, either, or none of the callback types. Both types are described below.
The first type is known as a “set parameter” callback, and can be installed by calling add_on_set_parameters_callback
.
The callback should accept a list of Parameter
objects, and return an rcl_interfaces/msg/SetParametersResult
.
This callback will be called before a parameter is declared or changed on a node.
The main purpose of this callback is to give the user the ability to inspect the upcoming change to the parameter and explicitly reject the change.
Note
It is important that “set parameter” callbacks have no side-effects. Since multiple “set parameter” callbacks can be chained, there is no way for an individual callback to know if a later callback will reject the update. If the individual callback were to make changes to the class it is in, for instance, it may get out-of-sync with the actual parameter. To get a callback after a parameter has been successfully changed, see the next type of callback below.
The second type of callback is known as an “on parameter event” callback, and can be installed by calling on_parameter_event
.
The callback should accept an rcl_interfaces/msg/ParameterEvent
object, and return nothing.
This callback will be called after all parameters have been declared, changed, or deleted.
The main purpose of this callback is to give the user the ability to react to changes from parameters that have successfully been accepted.
Interacting with parameters
ROS 2 nodes can perform parameter operations through node APIs as described in Using parameters in a class (C++) or Using parameters in a class (Python). External processes can perform parameter operations via parameter services that are created by default when a node is instantiated. The services that are created by default are:
/node_name/describe_parameters: Uses a service type of
rcl_interfaces/srv/DescribeParameters
. Given a list of parameter names, returns a list of descriptors associated with the parameters./node_name/get_parameter_types: Uses a service type of
rcl_interfaces/srv/GetParameterTypes
. Given a list of parameter names, returns a list of parameter types associated with the parameters./node_name/get_parameters: Uses a service type of
rcl_interfaces/srv/GetParameters
. Given a list of parameter names, returns a list of parameter values associated with the parameters./node_name/list_parameters: Uses a service type of
rcl_interfaces/srv/ListParameters
. Given an optional list of parameter prefixes, returns a list of the available parameters with that prefix. If the prefixes are empty, returns all parameters./node_name/set_parameters: Uses a service type of
rcl_interfaces/srv/SetParameters
. Given a list of parameter names and values, attempts to set the parameters on the node. Returns a list of results from trying to set each parameter; some of them may have succeeded and some may have failed./node_name/set_parameters_atomically: Uses a service type of
rcl_interfaces/srv/SetParametersAtomically
. Given a list of parameter names and values, attempts to set the parameters on the node. Returns a single result from trying to set all parameters, so if one failed, all of them failed.
Setting initial parameter values when running a node
Initial parameter values can be set when running the node either through individual command-line arguments, or through YAML files. See Setting parameters directly from the command line for examples on how to set initial parameter values.
Setting initial parameter values when launching nodes
Initial parameter values can also be set when running the node through the ROS 2 launch facility. See this document for information on how to specify parameters via launch.
Manipulating parameter values at runtime
The ros2 param
command is the general way to interact with parameters for nodes that are already running.
ros2 param
uses the parameter service API as described above to perform the various operations.
See the how-to guide for details on how to use ros2 param
.
Migrating from ROS 1
The Launch file migration guide explains how to migrate param
and rosparam
launch tags from ROS 1 to ROS 2.
The YAML parameter file migration guide explains how to migrate parameter files from ROS 1 to ROS 2.
In ROS 1, the roscore
acted like a global parameter blackboard where all nodes could get and set parameters.
Since there is no central roscore
in ROS 2, that functionality no longer exists.
The recommended approach in ROS 2 is to use per-node parameters that are closely tied to the nodes that use them.
If a global blackboard is still needed, it is possible to create a dedicated node for this purpose.
ROS 2 ships with one in in the ros-humble-demo-nodes-cpp
package called parameter_blackboard
; it can be run with:
ros2 run demo_nodes_cpp parameter_blackboard
The code for the parameter_blackboard
is here.