Skip to content

ROS2 parameter server which caches parameter into the storage if necessary and reload it during initialization.

License

Notifications You must be signed in to change notification settings

fujitatomoya/ros2_persist_parameter_server

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

42 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

humble jazzy rolling humble-nightly jazzy-nightly rolling-nightly

ROS2 Persistent Parameter Server

ROS 2 Persistent Parameter Server, that resides in the ROS 2 system to serve the parameter daemon. The other nodes(e.g the client demo provided in the code) can write/read the parameter in Parameter Server, and Parameter Server is able to store the parameter into the persistent storage which user can specify such as tmpfs, nfs, or disk.

See overview slide deck for general information.

Background

The discussion is opened here, and centralized parameter server is not a good affinity to ROS 2 distributed system architecture. One of the most valuable things about ROS APIs is that we make sure that the messages have specific semantic meaning so that they can’t be misinterpreted. As we develop the ROS 2 tools and best practices we should make sure to bring that same level of rigor to parameters too for greater reusability and correctness.

Although, it is expected to be the following requirement.

  • Global configuration that many nodes share (e.g. RTOS priorities, vehicle dimensions, …)
  • Generic ROS 2 system property server.
  • Persistent storage support to re-initialize the system. parameters are modified in runtime and cache it into persistent volume as well. and next boot or next re-spawn, modified parameter will be loaded at initialization. (parameter lifetime is dependent on use case, sometimes system lifetime, sometimes node lifetime.)
  • Using ROS1 based application with Parameter Server.

Overview

overview_architecture

Generally ROS 2 Parameter Server is simple blackboard to write/read parameters on that. The other nodes can write/read the parameter on the server to share them in the ROS 2 system. there is a new concept for "Persistent Parameter" which is described later.

ROS 2 Parameter Server is constructed on ROS parameter API's, nothing specific API's are provided to connect to the server from the client. Also, about the security it just relies on ROS 2 security aspect.

Persistent Parameter Registration

Persistent Prefix

persistent parameter must have prefix "persistent"

Scope Overview

parameter server has the following scope for persistent parameter. since parameter server is built on top of ROS 2 Parameter API, parameter server supports "persistent" parameter based on /parameter_events topic.

Category Supported Description
Parameter API YES ROS 2 Parameter Client API supported, since this activity can be detected via /parameter_events.
Persistent Parameter File YES parameter server dedicated argument to specify the file to load as parameters. in addition, all of the persistent parameters will be stored into this file during shutdown.
e.g) --file-path /tmp/parameter_server.yaml
Parameter Arguments NO e.g) --ros-args -p persistent.some_int:=42
some_int cannot be registed as persistent parameter, since this cannot be notified via /parameter_events to parameter server.
Parameter File Arguments NO e.g) --ros-args --params-file ./parameters_via_cli.yaml
same with parameter arguments, cannot be registered as persistent parameter, since these cannot be notified via /parameter_events to parameter server.
Launch Parameter NO e.g) ros2 launch parameter_server parameter_server.launch.py
same with parameter arguments, cannot be registered as persistent parameter, since these cannot be notified via /parameter_events to parameter server.

Configurable Options

  • Node Name Since ROS 2 parameter is owned by node, node name will be needed to access the parameters, this is designed to clearify semantics for the parameters and owners. Node name will be "parameter_server" if node name is not specifies. so the other nodes can use "parameter_server" as well to access in the same system Parameter Server. If there must exist multiple parameter servers, these parameter servers need to specify a different node name, such as "parameter_server_[special_string]", please notice that ROS 2 node name can only contains alphanumerics and '_'.
  • Persistent Volume Definition of "Persistent" is different from user and use cases, so it should be configurable to set the path to store the persistent --file-path FILE_PATH parameter. Expecting if the parameter's lifespan is system boot, path would be "/tmp" because user wants a fresh start via reboot. Or maybe physical persistent volume is chosen if users want to keep the parameter into the hardware storage. At the initialization time, Parameter Server will load the parameters from the storage which is specified by user.
  • Node Options there are two important options, allow_undeclared_parameters: (default true) automatically_declare_parameters_from_overrides: (default true)

all of the configuration options will be passed via arguments as followings.

Options CLI Description
Node Name --ros-args --remap __node:=NODENAME in default, "parameter_server" will be used.
Help --help show usage.
File Path --file-path FILE_PATH in default, "/tmp/parameter_server.yaml" will be used. if specified, that path will be used to store/load the parameter yaml file.
Node Options --allow-declare true/false default enabled, if specified allow any parameter name to be set on parameter server without declaration by itself. Otherwise it does not.
--allow-override true/false default enabled, if specified true iterate through the node's parameter overrides or implicitly declare any that have not already been declared.

Sequence

  1. parameter server is initialized via __params:=<xxx.yaml> this is just a initial parameter(not persistent) to load into the parameter server's memory.
  2. parameter server loads parameter specified yaml file via --file-path. and then parameter server will overwrite or declare parameters. (*) at #1 parameters might be overwritten.
  3. parameter server starts the main loop with callback for parameter changes.
  4. if the parameter changes are on "/persistent" that will be stored in storage at this time.
  5. at the finalization, flash all of the "/persistent" parameters into the file system.

Getting Started

Supported Distribution

Docker Container

see available images for tomoyafujita/ros2_param_server@dockerhub

docker run -it --rm --net=host tomoyafujita/ros2_param_server:humble

Dependent Packages

apt install libyaml-cpp-dev libboost-program-options-dev libboost-filesystem-dev

Prerequisites

ros2 source build environment(Linux-Development-Setup/) is required to build and run the parameter server.

Build

to install local colcon workspace,

# cd <colcon_workspace>/src
# git clone https://github.com/fujitatomoya/ros2_persist_parameter_server
# cd <colcon_workspace>
# colcon build --symlink-install --packages-select parameter_server ros2_persistent_parameter_server_test
# source install/local_setup.bash

Run

  1. start parameter server.

    # cp <colcon_workspace>/src/ros2_persist_parameter_server/server/param/parameter_server.yaml /tmp/
    # ros2 run parameter_server server
    [INFO] [parameter_server]: Parameter Server node named: '/parameter_server' started and ready, and serving '9' parameters already!
    ...<snip>
  2. update persistent parameter.

    # ros2 param set /parameter_server persistent.some_int 81
    Set parameter successful
    # ros2 param set /parameter_server persistent.a_string Konnichiwa
    Set parameter successful
    # ros2 param set /parameter_server persistent.pi 3.14159265359
    Set parameter successful
    # ros2 param set /parameter_server persistent.some_lists.some_integers 81,82,83,84
    Set parameter successful
  3. restart parameter server.

    # ros2 run parameter_server server
    [INFO] [parameter_server]: Parameter Server node named: '/parameter_server' started and ready, and serving '9' parameters already!
    ...<snip>
  4. check persistent parameter is precisely cached and loaded into parameter server.

    # ros2 param get /parameter_server persistent.a_string
    String value is: Konnichiwa
    # ros2 param get /parameter_server persistent.pi
    Double value is: 3.14159265359
    # ros2 param get /parameter_server persistent.some_int
    Integer value is: 81
    # ros2 param get /parameter_server persistent.some_lists.some_integers
    String value is: 81,82,83,84

Test

These samples verify the following functions.

  • persistent parameter can be read/stored to/from the file system.
  • persistent parameter can be read/modified from parameter client.
  • non-persistent parameter cannot be read/stored to/from the file system.
  • non-persistent parameter can be read/modified from parameter client

make sure to add the path of launch package to the PATH environment.

# source <launch_workspace>/install/setup.bash

Run

test.py is the entry for test.

test.py will call test.launch.py file to start persistent parameter server and the test client, it also creates a thread to kill parameter server after specified time. All function tests are finished in client.

!!!NOTE The test script will load the yaml file that should existed in /tmp/test, therefore, before executing test demo, you need to copy the yaml file existing in server directory to /tmp/test.

# mkdir -p /tmp/test
# cp <colcon_workspace>/src/ros2_persist_parameter/server/param/parameter_server.yaml /tmp/test
# ./<colcon_workspace>/src/ros2_persist_parameter/test/test.py

All of the test is listed with result as following

!!!NOTE Client has a 5-seconds sleep during server restarts.

......   // omit some output logs

[ros2-2] [INFO] [1601447662.145760479] [client]: ***************************************************************************
[ros2-2] [INFO] [1601447662.145794365] [client]: *********************************Test Result*******************************
[ros2-2] [INFO] [1601447662.145817265] [client]: a. Read Normal Parameter                                     :             PASS
[ros2-2] [INFO] [1601447662.145842530] [client]: b. Read Persistent Parameter                                 :             PASS
[ros2-2] [INFO] [1601447662.145863430] [client]: c. Modify Existed Normal parameter                           :             PASS
[ros2-2] [INFO] [1601447662.145885082] [client]: d. Modify Existed Persistent parameter                       :             PASS
[ros2-2] [INFO] [1601447662.145906067] [client]: e. Add New Normal parameter                                  :             PASS
[ros2-2] [INFO] [1601447662.145926790] [client]: f. Add New Persistent parameter                              :             PASS
[ros2-2] [INFO] [1601447662.145948146] [client]: g. Test Normal Parameter Not Stores To File                  :             PASS
[ros2-2] [INFO] [1601447662.145969623] [client]: h. Test Persistent Parameter Stores To File                  :             PASS
[ros2-2] [INFO] [1601447662.145990707] [client]: i. Test New Added Normal Parameter Not Stores To File        :             PASS
[ros2-2] [INFO] [1601447662.146011312] [client]: j. Test New Added Persistent Parameter Stores To File        :             PASS

Known Issues

  • Error when loading a persistent parameter of type array

    • [1.0,1.1] is deduced as [1, 1.1000000000000001] because of yaml-cpp bug, this will leads to Failed to parse parameters exception when loading the persistent parameters from yaml file.
    • The work-around is to use string sequence ["1.0", "1.1"] instead of double type, then change it into float() in the program.
  • Signal2(2) needs to be injected to the server executable

    • Because of ros2/ros2cli#899 and What is main process in container, signal (SIGINT/SIGTERM) does not directly go to the server process. This causes the server not to store the persistent parameters in the file system, since the server expects the signal to shutdown the process and store the all persistent parameters in the specified file system.
    • The work-around is that, configure container main process with the server executables (not using ros2 run until ros2/ros2cli#899 is solved) or send the signal from the host system to the server process in the container using docker exec <YOUR CONTAINER> kill -SIGINT <SERVER PID IN CONTAINER>.

Authors

License

Apache 2.0

About

ROS2 parameter server which caches parameter into the storage if necessary and reload it during initialization.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •