Skip to content

Commit

Permalink
Add throttle option for logging (PR #14)
Browse files Browse the repository at this point in the history
  • Loading branch information
David Conner committed May 1, 2024
1 parent de06e21 commit 4b79bb0
Show file tree
Hide file tree
Showing 3 changed files with 413 additions and 15 deletions.
99 changes: 84 additions & 15 deletions flexbe_core/flexbe_core/logger.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
#!/usr/bin/env python

# Copyright 2023 Philipp Schillinger, Team ViGIR, Christopher Newport University
# Copyright 2024 Philipp Schillinger, Team ViGIR, Christopher Newport University
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
Expand Down Expand Up @@ -31,7 +31,9 @@

"""Realize behavior-specific logging."""

from rclpy.exceptions import ParameterNotDeclaredException
from rclpy.node import Node
from rclpy.duration import Duration

from flexbe_msgs.msg import BehaviorLog

Expand All @@ -47,13 +49,37 @@ class Logger:

LOGGING_TOPIC = 'flexbe/log'

# max number of items in last logged dict (used for log_throttle)
MAX_LAST_LOGGED_SIZE = 1024
LAST_LOGGED_CLEARING_RATIO = 0.2

_pub = None
_node = None

@staticmethod
def initialize(node: Node):
Logger._node = node
Logger._pub = node.create_publisher(BehaviorLog, Logger.LOGGING_TOPIC, 100)
Logger._last_logged = {}

# Optional parameters that can be defined
try:
size_param = node.get_parameter("max_throttle_logging_size")
if size_param.type_ == size_param.Type.INTEGER:
Logger.MAX_LAST_LOGGED_SIZE = size_param.value
except ParameterNotDeclaredException as exc:
pass

try:
clear_param = node.get_parameter("throttle_logging_clear_ratio")
if clear_param.type_ in (clear_param.Type.INTEGER, clear_param.Type.DOUBLE):
Logger.LAST_LOGGED_CLEARING_RATIO = clear_param.value
except ParameterNotDeclaredException as exc:
pass

Logger._node.get_logger().debug(f"Enable throttle logging option with "
f"max size={Logger.MAX_LAST_LOGGED_SIZE} "
f"clear ratio={Logger.LAST_LOGGED_CLEARING_RATIO}")

@staticmethod
def log(text: str, severity: int):
Expand All @@ -67,6 +93,25 @@ def log(text: str, severity: int):
# also log locally
Logger.local(text, severity)

@staticmethod
def log_throttle(period : float, text: str, severity : int):
# create unique identifier for each logging message
log_id = f"{severity}_{text}"
time_now = Logger._node.get_clock().now()
# only log when it's the first time or period time has passed for the logging message
if not log_id in Logger._last_logged.keys() or \
time_now - Logger._last_logged[log_id] > Duration(seconds=period):
Logger.log(text, severity)
Logger._last_logged.update({log_id: time_now})

if len(Logger._last_logged) > Logger.MAX_LAST_LOGGED_SIZE:
# iterate through last logged items, sorted by the timestamp (oldest last)
clear_size = Logger.MAX_LAST_LOGGED_SIZE * (1 - Logger.LAST_LOGGED_CLEARING_RATIO)
for i, log in enumerate(sorted(Logger._last_logged.items(), key=lambda item: item[1], reverse=True)):
# remove defined percentage of oldest items
if i > clear_size:
Logger._last_logged.pop(log[0])

@staticmethod
def local(text: str, severity: int):
if Logger._node is None:
Expand All @@ -87,57 +132,81 @@ def local(text: str, severity: int):
# NOTE: Below text strings can only have single % symbols if they are being treated
# as format strings with appropriate arguments (otherwise replace with %% for simple string without args)
@staticmethod
def logdebug(text, *args):
def logdebug(text: str, *args):
Logger.log(text % args, Logger.REPORT_DEBUG)

@staticmethod
def loginfo(text, *args):
def loginfo(text: str, *args):
Logger.log(text % args, Logger.REPORT_INFO)

@staticmethod
def logwarn(text, *args):
def logwarn(text: str, *args):
Logger.log(text % args, Logger.REPORT_WARN)

@staticmethod
def loghint(text, *args):
def loghint(text: str, *args):
Logger.log(text % args, Logger.REPORT_HINT)

@staticmethod
def logerr(text, *args):
def logerr(text: str, *args):
Logger.log(text % args, Logger.REPORT_ERROR)

@staticmethod
def localdebug(text, *args):
def logdebug_throttle(period: float, text: str, *args):
Logger.log_throttle(period, text % args, Logger.REPORT_DEBUG)

@staticmethod
def loginfo_throttle(period: float, text: str, *args):
Logger.log_throttle(period, text % args, Logger.REPORT_INFO)

@staticmethod
def logwarn_throttle(period: float, text: str, *args):
Logger.log_throttle(period, text % args, Logger.REPORT_WARN)

@staticmethod
def loghint_throttle(period: float, text: str, *args):
Logger.log_throttle(period, text % args, Logger.REPORT_HINT)

@staticmethod
def logerr_throttle(period: float, text: str, *args):
Logger.log_throttle(period, text % args, Logger.REPORT_ERROR)

@staticmethod
def localdebug(text: str, *args):
Logger.local(text % args, Logger.REPORT_DEBUG)

@staticmethod
def localinfo(text, *args):
def localinfo(text: str, *args):
Logger.local(text % args, Logger.REPORT_INFO)

@staticmethod
def localwarn(text, *args):
def localwarn(text: str, *args):
Logger.local(text % args, Logger.REPORT_WARN)

@staticmethod
def localerr(text, *args):
def localhint(text: str, *args):
Logger.local(text % args, Logger.REPORT_HINT)

@staticmethod
def localerr(text: str, *args):
Logger.local(text % args, Logger.REPORT_ERROR)

@staticmethod
def debug(text, *args):
def debug(text: str, *args):
Logger.logdebug(text, *args)

@staticmethod
def info(text, *args):
def info(text: str, *args):
Logger.loginfo(text, *args)

@staticmethod
def warning(text, *args):
def warning(text: str, *args):
Logger.logwarn(text, *args)

@staticmethod
def hint(text, *args):
def hint(text: str, *args):
Logger.loghint(text, *args)

@staticmethod
def error(text, *args):
def error(text: str, *args):
Logger.logerr(text, *args)
67 changes: 67 additions & 0 deletions flexbe_core/test/flexbe_logger_test.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,67 @@
#!/usr/bin/env python3

# Copyright 2024 Christopher Newport University
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
#
# * Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
#
# * Neither the name of the Philipp Schillinger, Team ViGIR, Christopher Newport University nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.


"""Test description for test proxies."""
import os
import sys
import launch
import launch_testing.actions
import pytest


@pytest.mark.rostest
def generate_test_description():

path_to_test = os.path.dirname(__file__)

TEST_PROC_PATH = os.path.join(path_to_test, 'test_logger.py')

# This is necessary to get unbuffered output from the process under test
proc_env = os.environ.copy()
proc_env['PYTHONUNBUFFERED'] = '1'

test_logger = launch.actions.ExecuteProcess(
cmd=[sys.executable, TEST_PROC_PATH],
env=proc_env,
output='screen',
sigterm_timeout=launch.substitutions.LaunchConfiguration('sigterm_timeout', default=90),
sigkill_timeout=launch.substitutions.LaunchConfiguration('sigkill_timeout', default=90)
)

return (
launch.LaunchDescription([
test_logger,
launch_testing.actions.ReadyToTest()
]),
{
'test_logger': test_logger,
}
)
Loading

0 comments on commit 4b79bb0

Please sign in to comment.