Skip to content

Latest commit

 

History

History
518 lines (362 loc) · 18 KB

README.md

File metadata and controls

518 lines (362 loc) · 18 KB

Autopsy

Python utilities for the F1tenth project.

Currently, this package contains following modules:

Reconfigure module

This module is used to make dynamic reconfiguration for our nodes much easier.

Background

Dynamic reconfiguration is a great tool of ROS, using which you can adjust parameters of nodes real-time.

In ROS this feature is created by package dynamic_reconfigure. In case you would like to learn how to create a dynamic reconfiguration support, see their tutorial.

Unfortunately, usage of this feature is really troublesome. Especially, when combined with Python, since:

  • It is compiled.
  • You need to know the variables and their limit before running the node.
  • The syntax is really hard.

Because of this, there is an enhanced version called ddynamic_reconfigure that allows to modify the parameters using the dynamic reconfigure framework without having to write configuration files. It is written in C++ and used, e.g., in RealSense package.

To make the whole process easier, we have created our own Py script for dynamic reconfiguration support.

The GUI for this remains the same, i.e., rqt_reconfigure.

Usage

Import

from autopsy.reconfigure import ParameterServer

# Create an instance of the ParameterServer
P = ParameterServer()

Parameter definition

Parameters are 'properties' of ParameterServer.

We currently support:

  • integers
  • floats
  • strings
  • booleans
  • enums

Upon assigning a value to parameter, an internal object is created.

  • Mandatory
    • name -- name of the parameter, set to the name of the parameter
    • default -- default value of the parameter, set to the assigned value
  • Optional
    • description -- short comment for the parameter, str
    • value -- current value of the parameter, same as type
    • min -- minimum value for the parameter, int/float only
    • max -- maximum value for the parameter, int/float only
    • callback -- function to be called upon receiving parameter change, callable

Accessing the value can be done using .value. However, the operators are overloaded, meaning that you can freely use P.something + 5 without .value.

Note: The .value is used only when you want to store it somewhere else.

Note: Feature is are not currently available here.

Example:

# 1) Create an int named 'something' with value 5:
P.something = 5

# 2) Add some constraints/etc. for 'something':
P.something.description = "This is just a number"
P.something.min = 2
P.something.max = 6
# Note: Storing different value than min<=X<=max raises an exception.

# 3) Change the value in 'something':
P.something = 4

# 4) Create parameters of other types:
P.floatparam = 4.5
P.juststring = "I am a string."
P.boolean = True

# 5) Set limits right during creation:
P.anotherparam = {'default': 5.2, 'min': 2, 'max': 6, 'description': 'I am just a float.'}

# 6) Convert dynamic_reconfiguration cfg
# Instead of:
# gen.add("Speed_max", double_t, 0, "Level-1 speed", 0.158483034, 0, 0.3)
# do this:
P.smax = ["Speed_max", float, 0, "Level-1 speed", 0.158483034, 0, 0.3]

Parameters may be also defined at once using function P.update(PARAMETERS). There are two ways to use it:

  • Using dictionary (parameters are ordered randomly):
    PARAMETERS = {
        "something": 4,
        "anotherparam": {'default': 5.2, 'min': 2, 'max': 6, 'description': 'I am just a float.'},
    }
  • Using list of tuples (parameters are ordered):
    PARAMETERS = [
        ("something", 4),
        ("anotherparam", {'default': 5.2, 'min': 2, 'max': 6, 'description': 'I am just a float.'}),
    ]
Enumerated values

Example:

# 7) Enumerated parameters
from autopsy.reconfigure import ParameterEnum

class standardEnum(Enum):
    Value1 = 5
    Value2 = 10

class extendedEnum(ParameterEnum):
    Value1 = 9.5, "First value"
    Value2 = 9.68, "Second value"

inlineEnum = ParameterEnum("NOT_USED", {"Value1": (8, "with help"), "Value2": 9})

P.enumeratedParam = standardEnum
P.enumeratedParam2 = extendedEnum
P.enumeratedParam3 = inlineEnum

# Value assignment
P.enumeratedParam = standardEnum.Value1
Callbacks

Callback is a function that is called upon receiving a changed value of the parameter from the reconfigure service.

This function has to:

  • Have at least one argument, that contains the new value of the parameter.
  • Return value that is correct, i.e., that is sent back to the reconfigure server.

Example:

P = ParameterServer()
P.number = 5

def limit_value(new_value):
    "Limits the value to be lower then 10."

    if new_value <= 10:
        value = new_value
    else:
        value = P.number.value

    return value

P.number.callback = limit_value

P.reconfigure()

# limit_value is run on every change of 'P.number'.

Note: Callback function receives new value of the parameter, and is required to return the true new value. It is then filled inside the parameter and announced to the reconfigure GUI.

Linked variables

Parameters of type ConstrainedP can be "linked" together. When done, first parameter cannot exceed the value of the second parameter and vice versa. Linking is performed after defining the parameters using P.link().

Example:

P = ParameterServer()

P.range_min = 2.0
P.range_max = 3.0

P.link(P.range_min, P.range_max)

# P.range_min and P.range_max are now linked. Should min be over max, it is set to max, and vice versa.

Reconfiguration itself

# 1) Update parameters (with values in the ROS Parameter Server)
# This is a "mass update", just dict of values
if rospy.has_param("~"):
    P.update(rospy.get_param("~"))

# 2) Current values of Parameters
print (P)
print (P.something)

# 3) Allow reconfiguration
# Call this after 'init_node'. It will create the callbacks and so on.
P.reconfigure()

Note: Passing a string to P.reconfigure() changes the namespace of the ParameterServer in ROS.

Note: From >0.5.1 an additional bool can be passed to P.update() to only update existing parameters.

Compatibility

In order to make autopsy.reconfiguration optional, you can use following construct:

try:
    from autopsy.reconfigure import ParameterServer
    AUTOPSY_AVAILABLE = True

except:
    class ParameterServer(dict):
        """dot.notation access to dictionary attributes

        Source:
        https://stackoverflow.com/questions/2352181/how-to-use-a-dot-to-access-members-of-dictionary
        """
        __getattr__ = dict.get
        __setattr__ = dict.__setitem__
        __delattr__ = dict.__delitem__

    AUTOPSY_AVAILABLE = False

This is effective for >0.4.0. Just do not use .value at all.

Until 0.4.0

To make your code runnable on <= 0.4.0 stick to these rules:

  • P.reconfigure() has no argument.
  • Everytime you access the parameter value use .value.

ROS 2

To use reconfiguration utility in ROS2, you have to pass the Node instance inside node argument:

P.reconfigure(node = node)

In the current implementation following features are not supported:

  • enums,
  • linked variables.

Because of the limitations of ROS2 API it is most likely that they will be never supported.

As from the ROS2 Parameter API side these features are currently not implemented:

  • IntegerRange (limits are supported, step is not),
  • FloatingPointRange (limits are supported, step is not),
  • Arrays.

Full example

#!/usr/bin/env python
# reconfiguration_example.py

import rospy
from autopsy.reconfigure import ParameterServer

P = ParameterServer()

P.value = 5

rospy.init_node("reconfiguration_example")

rate = rospy.Rate(1)

if rospy.has_param("~"):
    P.update(rospy.get_param("~"))

print ("Current values of the parameters:")
print (P)

P.reconfigure()

while not rospy.is_shutdown():
    print ("Current value:", P.value)
    rate.sleep()

Uninode module

Universal node class with ROS 1 / ROS 2 support.

Universal node (or uninode) serves as a compatibility layer for a ROS node to be executable from both ROS versions. Basically, for ROS2 we use mostly the original 'rclpy.node.Node', whereas for ROS1 we try to match its functions onto this class.

Relations

ROS 1 uninode ROS 2
rospy.init_node self.__init__ self.__init__
rospy.get_name self.get_name self.get_name
rospy.get_time self.get_time+ using self.get_clock()
rospy.Publisher self.Publisher+ self.create_publisher
rospy.Subscriber self.Subscriber+ self.create_subscription
rospy.Rate self.Rate+ self.create_rate
rospy.Timer self.Timer+ self.create_timer
rospy.Service self.Service+ self.create_service
rospy.ServiceProxy self.ServiceProxy+ self.create_client
rospy.Time.now self.Time.now+ self.get_clock().now()
rospy.logdebug self.logdebug+ self.get_logger().debug
rospy.loginfo self.loginfo+ self.get_logger().info
rospy.logwarn self.logwarn+ self.get_logger().warn
rospy.logerr self.logerr+ self.get_logger().error
rospy.logerror self.logerror+ self.get_logger().error
rospy.logfatal self.logfatal+ self.get_logger().fatal

Note: Lines with '+' denote that the same function as for ROS2 can be used for uninode.

Differences

  • ROS1
  • ROS2
    • Timer in ROS2 does not take any arguments (in contrast to the 'rospy.TimerEvent in ROS1). Therefore, the function has to be created as: def callback(self, *args, **kwargs)
    • There is no direct equivalent for rospy.get_time() in ROS2. However, it can be obtained using self.get_clock().now().nanoseconds * (10 ** 9).
  • Common
    • Services are handled slightly differently in both ROS versions. At first, service message is compiled into two in ROS1. In ROS2 there is only one message type. In addition, service callback has only one argument in ROS1, whereas there are two arguments in ROS2 (second is the response). To make it compatible with both versions stick to this:
      if autopsy.node.ROS_VERSION == 1:
          from package.srv import ResponseMessage
      
      def callback(self, msg, response = None):
          ...
          if response is None:
              return ResponseMessage(...)
          else:
              response.data = ...
              return response

Full example

Example:

from autopsy.node import Node
from std_msgs.msg import Int32

class MyNode(Node):

    def __init__(self, name):
        super(Node, self).__init__(name)

        self._subscriber = self.Subscriber("/topic", Int32, self.topic_callback)


    def topic_callback(self, msg):
        print ("Received:", msg.data)

Unicore module

Universal core layer for both ROS 1 and ROS 2.

Universal core (or unicore) serves as a compatibility layer for running a ROS node that is executable from both ROS versions.

Features

Here follows a list of implemented functions from 'rospy'/'rclpy':

  • (ROS2) init()
  • spin()
  • (ROS2) spin_once()
  • (ROS2) spin_until_future_complete()
  • (ROS2) shutdown()
  • duration.Duration() -- Duration is used to store stamps inside the messages. The opposite action is not yet implemented.

Full example

Example:

from autopsy.core import *
from autopsy.node import Node
from std_msgs.msg import Int32

@ros2_only(other_callback)
def callback(data):
    print("Called from ROS2!")

def other_callback(data):
    print("Called from ROS1!")

Core.init()

n = Node("testing_node")
n.Subscriber("/topic", Int32, callback)

Core.spin(n)

Time module

Set of utilities for measuring duration of code blocks. This was originally a part of rosmeasure package.

Classes

The autopsy.time utility currently provides following classes:

  • TimeMeasurer

TimeMeasurer

TimeMeasurer measures time spend in the selected section of the code. It is created as:

tm = TimeMeasurer(
    name = "Measurer",  # Name of the Measurer
    units = "s"         # Time units used for the measuring
)

Upon creation, these functions are used:

  • start() -- Starts the measurement.
  • end() -- Ends the measurement, storing the values inside.
  • summary() -- Prints out the statistics for the Measurer.

Another way of using this measurer is as follows:

with TimeMeasurer(name, units) as _ :
    ...

Decorators

The autopsy.time utility also provides decorators to be used instead of the classes:

  • @duration

@duration

Duration decorator is basically the same as TimeMeasurer. It is used as follows:

@duration(name, units)
def function():
    pass

The decorator supplies following section of the code:

TM = TimeMeasurer(name, units)
TM.start()
output = function()
TM.end()
TM.summary()
return output

However, in contrast to TimeMeasurer, a keyword-only argument interval can be passed to the decorator to report the summary periodically, and not on every function call.

Since >0.10.1 not passing interval generates a warning on start-up.