1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00

Merge pull request #11573 from aabadie/pr/sys/log_color

sys/log: add module for colorized logging
This commit is contained in:
Alexandre Abadie 2019-09-09 21:02:00 +02:00 committed by GitHub
commit 60229d1f15
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 209 additions and 0 deletions

View File

@ -665,6 +665,14 @@ class SerCmd(cmd.Cmd):
else: else:
output += c output += c
# Hack to correctly handle reset ANSI escape code in the stream
# When the reset escape sequence is detected, it is written and
# flushed immediately to stdout
if output == '\033[0m':
sys.stdout.write(output)
sys.stdout.flush()
output = ""
crreceived = c == '\r' crreceived = c == '\r'
nlreceived = c == '\n' nlreceived = c == '\n'

View File

@ -44,6 +44,7 @@ PSEUDOMODULES += l2filter_whitelist
PSEUDOMODULES += lis2dh12_spi PSEUDOMODULES += lis2dh12_spi
PSEUDOMODULES += log PSEUDOMODULES += log
PSEUDOMODULES += log_printfnoformat PSEUDOMODULES += log_printfnoformat
PSEUDOMODULES += log_color
PSEUDOMODULES += lora PSEUDOMODULES += lora
PSEUDOMODULES += mpu_stack_guard PSEUDOMODULES += mpu_stack_guard
PSEUDOMODULES += nanocoap_% PSEUDOMODULES += nanocoap_%

View File

@ -1,3 +1,7 @@
ifneq (,$(filter log_printfnoformat,$(USEMODULE))) ifneq (,$(filter log_printfnoformat,$(USEMODULE)))
USEMODULE_INCLUDES += $(RIOTBASE)/sys/log/log_printfnoformat USEMODULE_INCLUDES += $(RIOTBASE)/sys/log/log_printfnoformat
endif endif
ifneq (,$(filter log_color,$(USEMODULE)))
USEMODULE_INCLUDES += $(RIOTBASE)/sys/log/log_color
endif

View File

@ -0,0 +1,123 @@
/*
* Copyright (C) 2019 Inria
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @defgroup sys_log_color Colored log module
* @ingroup sys
* @brief This module implements a logging module with colored output
* @{
*
* @file
* @brief log_module header
*
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
*/
#ifndef LOG_MODULE_H
#define LOG_MODULE_H
#include <assert.h>
#include <stdio.h>
#include <stdarg.h>
#include "log.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief Default ANSI color escape code for error logs
*
* Default is bold red
*/
#ifndef LOG_ERROR_ANSI_COLOR_CODE
#define LOG_ERROR_ANSI_COLOR_CODE ("\033[1;31m")
#endif
/**
* @brief Default ANSI color escape code for warning logs
*
* Default is bold yellow
*/
#ifndef LOG_WARNING_ANSI_COLOR_CODE
#define LOG_WARNING_ANSI_COLOR_CODE ("\033[1;33m")
#endif
/**
* @brief Default ANSI color escape code for info logs
*
* Default is bold white
*/
#ifndef LOG_INFO_ANSI_COLOR_CODE
#define LOG_INFO_ANSI_COLOR_CODE ("\033[1m")
#endif
/**
* @brief Default ANSI color escape code for debug logs
*
* Default is green
*/
#ifndef LOG_DEBUG_ANSI_COLOR_CODE
#define LOG_DEBUG_ANSI_COLOR_CODE ("\033[0;32m")
#endif
/**
* @brief ANSI color escape code used for resetting color
*/
#define LOG_RESET_ANSI_COLOR_CODE ("\033[0m")
/**
* @brief ANSI color escape codes array
*
* Internal use only
*/
static const char * const _ansi_codes[] =
{
[LOG_ERROR] = LOG_ERROR_ANSI_COLOR_CODE,
[LOG_WARNING] = LOG_WARNING_ANSI_COLOR_CODE,
[LOG_INFO] = LOG_INFO_ANSI_COLOR_CODE,
[LOG_DEBUG] = LOG_DEBUG_ANSI_COLOR_CODE,
};
/**
* @brief log_write overridden function for colored output
*
* @param[in] level Logging level
* @param[in] format String format to print
*/
static inline void log_write(unsigned level, const char *format, ...)
{
assert(level > 0);
printf("%s", _ansi_codes[level]);
va_list args;
va_start(args, format);
/* Temporarily disable clang format-nonliteral warning */
#ifdef __clang__
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wformat-nonliteral"
#endif /* clang */
vprintf(format, args);
#ifdef __clang__
#pragma clang diagnostic pop
#endif /* clang */
va_end(args);
printf(LOG_RESET_ANSI_COLOR_CODE);
#ifdef MODULE_NEWLIB
/* no fflush on msp430 */
fflush(stdout);
#endif
}
#ifdef __cplusplus
}
#endif
/**@}*/
#endif /* LOG_MODULE_H */

8
tests/log_color/Makefile Normal file
View File

@ -0,0 +1,8 @@
include ../Makefile.tests_common
USEMODULE += log_color
# Enable debug log level
CFLAGS += -DLOG_LEVEL=4
include $(RIOTBASE)/Makefile.include

33
tests/log_color/main.c Normal file
View File

@ -0,0 +1,33 @@
/*
* Copyright (C) 2019 Inria
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @file
* @brief Test logging with colors gives the expected output
*
* @author Alexandre Abadie <alexandre.abadie@inria.fr>
*
*/
#include <inttypes.h>
#include "log.h"
int main(void)
{
const uint8_t value = 42;
const char *string = "test";
const char *format = "Logging value '%d' and string '%s'\n";
LOG_ERROR(format, value, string);
LOG_WARNING(format, value, string);
LOG_INFO(format, value, string);
LOG_DEBUG(format, value, string);
return 0;
}

32
tests/log_color/tests/01-run.py Executable file
View File

@ -0,0 +1,32 @@
#!/usr/bin/env python3
# Copyright (C) 2019 Alexandre Abadie <alexandre.abadie@inria.fr>
#
# This file is subject to the terms and conditions of the GNU Lesser
# General Public License v2.1. See the file LICENSE in the top level
# directory for more details.
import sys
from testrunner import run
VALUE = 42
STRING = 'test'
STRING_FORMAT = '{}{}Logging value \'{}\' and string \'{}\''
ERROR = '\033[1;31m'
WARNING = '\033[1;33m'
INFO = '\033[1m'
DEBUG = '\033[0;32m'
RESET = '\033[0m'
LEVELS = [ERROR, WARNING, INFO, DEBUG]
def testfunc(child):
for level in LEVELS:
child.expect_exact(STRING_FORMAT.format(RESET, level, VALUE, STRING))
if __name__ == "__main__":
sys.exit(run(testfunc))