2018-07-04 15:39:44 +01:00
|
|
|
# Copyright 2013-2018 ARM Limited
|
2017-03-06 11:10:25 +00:00
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
#
|
|
|
|
|
|
|
|
|
|
|
|
# pylint: disable=E1101
|
|
|
|
import logging
|
2018-04-30 11:30:45 +01:00
|
|
|
import logging.handlers
|
|
|
|
import os
|
2017-03-06 11:10:25 +00:00
|
|
|
import string
|
|
|
|
import subprocess
|
2018-04-30 11:30:45 +01:00
|
|
|
import threading
|
2018-04-30 13:13:57 +01:00
|
|
|
from contextlib import contextmanager
|
2017-03-06 11:10:25 +00:00
|
|
|
|
|
|
|
import colorama
|
|
|
|
|
2017-03-09 17:39:44 +00:00
|
|
|
from devlib import DevlibError
|
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
from wa.framework import signal
|
|
|
|
from wa.framework.exception import WAError
|
|
|
|
from wa.utils.misc import get_traceback
|
|
|
|
|
|
|
|
|
|
|
|
COLOR_MAP = {
|
|
|
|
logging.DEBUG: colorama.Fore.BLUE,
|
|
|
|
logging.INFO: colorama.Fore.GREEN,
|
|
|
|
logging.WARNING: colorama.Fore.YELLOW,
|
|
|
|
logging.ERROR: colorama.Fore.RED,
|
|
|
|
logging.CRITICAL: colorama.Style.BRIGHT + colorama.Fore.RED,
|
|
|
|
}
|
|
|
|
|
|
|
|
RESET_COLOR = colorama.Style.RESET_ALL
|
|
|
|
|
2018-04-30 11:30:45 +01:00
|
|
|
DEFAULT_INIT_BUFFER_CAPACITY = 1000
|
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
_indent_level = 0
|
|
|
|
_indent_width = 4
|
|
|
|
_console_handler = None
|
2018-04-30 11:30:45 +01:00
|
|
|
_init_handler = None
|
2017-03-06 11:10:25 +00:00
|
|
|
|
2018-07-04 17:44:55 +01:00
|
|
|
|
|
|
|
# pylint: disable=global-statement
|
2017-03-06 11:10:25 +00:00
|
|
|
def init(verbosity=logging.INFO, color=True, indent_with=4,
|
|
|
|
regular_fmt='%(levelname)-8s %(message)s',
|
|
|
|
verbose_fmt='%(asctime)s %(levelname)-8s %(name)10.10s: %(message)s',
|
|
|
|
debug=False):
|
2018-04-30 11:30:45 +01:00
|
|
|
global _indent_width, _console_handler, _init_handler
|
2017-03-06 11:10:25 +00:00
|
|
|
_indent_width = indent_with
|
|
|
|
signal.log_error_func = lambda m: log_error(m, signal.logger)
|
|
|
|
|
|
|
|
root_logger = logging.getLogger()
|
|
|
|
root_logger.setLevel(logging.DEBUG)
|
|
|
|
|
|
|
|
error_handler = ErrorSignalHandler(logging.DEBUG)
|
|
|
|
root_logger.addHandler(error_handler)
|
|
|
|
|
|
|
|
_console_handler = logging.StreamHandler()
|
|
|
|
if color:
|
|
|
|
formatter = ColorFormatter
|
|
|
|
else:
|
|
|
|
formatter = LineFormatter
|
|
|
|
if verbosity:
|
|
|
|
_console_handler.setLevel(logging.DEBUG)
|
|
|
|
_console_handler.setFormatter(formatter(verbose_fmt))
|
|
|
|
else:
|
|
|
|
_console_handler.setLevel(logging.INFO)
|
|
|
|
_console_handler.setFormatter(formatter(regular_fmt))
|
|
|
|
root_logger.addHandler(_console_handler)
|
2018-04-30 11:30:45 +01:00
|
|
|
|
2020-10-19 18:07:21 +01:00
|
|
|
buffer_capacity = int(os.getenv('WA_LOG_BUFFER_CAPACITY',
|
2020-10-19 18:09:04 +01:00
|
|
|
str(DEFAULT_INIT_BUFFER_CAPACITY)))
|
2018-04-30 11:30:45 +01:00
|
|
|
_init_handler = InitHandler(buffer_capacity)
|
|
|
|
_init_handler.setLevel(logging.DEBUG)
|
|
|
|
root_logger.addHandler(_init_handler)
|
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
logging.basicConfig(level=logging.DEBUG)
|
|
|
|
if not debug:
|
|
|
|
logging.raiseExceptions = False
|
|
|
|
|
2017-10-05 16:48:54 +01:00
|
|
|
logger = logging.getLogger('CGroups')
|
|
|
|
logger.info = logger.debug
|
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
|
|
|
|
def set_level(level):
|
|
|
|
_console_handler.setLevel(level)
|
|
|
|
|
|
|
|
|
2018-07-04 17:44:55 +01:00
|
|
|
# pylint: disable=global-statement
|
2017-03-06 11:10:25 +00:00
|
|
|
def add_file(filepath, level=logging.DEBUG,
|
2017-03-06 11:10:25 +00:00
|
|
|
fmt='%(asctime)s %(levelname)-8s %(name)10.10s: %(message)s'):
|
2018-04-30 11:30:45 +01:00
|
|
|
global _init_handler
|
2017-03-06 11:10:25 +00:00
|
|
|
root_logger = logging.getLogger()
|
|
|
|
file_handler = logging.FileHandler(filepath)
|
|
|
|
file_handler.setLevel(level)
|
|
|
|
file_handler.setFormatter(LineFormatter(fmt))
|
2018-04-30 11:30:45 +01:00
|
|
|
|
|
|
|
if _init_handler:
|
|
|
|
_init_handler.flush_to_target(file_handler)
|
|
|
|
root_logger.removeHandler(_init_handler)
|
|
|
|
_init_handler = None
|
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
root_logger.addHandler(file_handler)
|
|
|
|
|
|
|
|
|
|
|
|
def enable(logs):
|
|
|
|
if isinstance(logs, list):
|
|
|
|
for log in logs:
|
|
|
|
__enable_logger(log)
|
|
|
|
else:
|
|
|
|
__enable_logger(logs)
|
|
|
|
|
|
|
|
|
|
|
|
def disable(logs):
|
|
|
|
if isinstance(logs, list):
|
|
|
|
for log in logs:
|
|
|
|
__disable_logger(log)
|
|
|
|
else:
|
|
|
|
__disable_logger(logs)
|
|
|
|
|
|
|
|
|
|
|
|
def __enable_logger(logger):
|
2018-05-30 13:58:49 +01:00
|
|
|
if isinstance(logger, str):
|
2017-03-06 11:10:25 +00:00
|
|
|
logger = logging.getLogger(logger)
|
|
|
|
logger.propagate = True
|
|
|
|
|
|
|
|
|
|
|
|
def __disable_logger(logger):
|
2018-05-30 13:58:49 +01:00
|
|
|
if isinstance(logger, str):
|
2017-03-06 11:10:25 +00:00
|
|
|
logger = logging.getLogger(logger)
|
|
|
|
logger.propagate = False
|
|
|
|
|
|
|
|
|
2018-07-04 17:44:55 +01:00
|
|
|
# pylint: disable=global-statement
|
2017-03-06 11:10:25 +00:00
|
|
|
def indent():
|
|
|
|
global _indent_level
|
|
|
|
_indent_level += 1
|
|
|
|
|
|
|
|
|
2018-07-04 17:44:55 +01:00
|
|
|
# pylint: disable=global-statement
|
2017-03-06 11:10:25 +00:00
|
|
|
def dedent():
|
|
|
|
global _indent_level
|
|
|
|
_indent_level -= 1
|
|
|
|
|
|
|
|
|
2018-04-30 13:13:57 +01:00
|
|
|
@contextmanager
|
|
|
|
def indentcontext():
|
|
|
|
indent()
|
|
|
|
try:
|
|
|
|
yield
|
|
|
|
finally:
|
|
|
|
dedent()
|
|
|
|
|
|
|
|
|
2018-07-04 17:44:55 +01:00
|
|
|
# pylint: disable=global-statement
|
2018-02-28 15:18:58 +00:00
|
|
|
def set_indent_level(level):
|
|
|
|
global _indent_level
|
|
|
|
old_level = _indent_level
|
|
|
|
_indent_level = level
|
|
|
|
return old_level
|
|
|
|
|
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
def log_error(e, logger, critical=False):
|
|
|
|
"""
|
|
|
|
Log the specified Exception as an error. The Error message will be formatted
|
|
|
|
differently depending on the nature of the exception.
|
|
|
|
|
|
|
|
:e: the error to log. should be an instance of ``Exception``
|
|
|
|
:logger: logger to be used.
|
2017-10-23 17:25:02 +01:00
|
|
|
:critical: if ``True``, this error will be logged at ``logging.CRITICAL``
|
2017-03-06 11:10:25 +00:00
|
|
|
level, otherwise it will be logged as ``logging.ERROR``.
|
2017-10-23 17:25:02 +01:00
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
"""
|
2018-02-28 15:18:58 +00:00
|
|
|
if getattr(e, 'logged', None):
|
|
|
|
return
|
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
if critical:
|
|
|
|
log_func = logger.critical
|
|
|
|
else:
|
|
|
|
log_func = logger.error
|
|
|
|
|
|
|
|
if isinstance(e, KeyboardInterrupt):
|
2018-02-28 15:18:58 +00:00
|
|
|
old_level = set_indent_level(0)
|
|
|
|
logger.info('Got CTRL-C. Aborting.')
|
|
|
|
set_indent_level(old_level)
|
2018-07-09 15:28:22 +01:00
|
|
|
elif isinstance(e, (WAError, DevlibError)):
|
2017-10-23 17:25:15 +01:00
|
|
|
log_func(str(e))
|
2017-03-06 11:10:25 +00:00
|
|
|
elif isinstance(e, subprocess.CalledProcessError):
|
|
|
|
tb = get_traceback()
|
|
|
|
log_func(tb)
|
|
|
|
command = e.cmd
|
|
|
|
if e.args:
|
2018-10-09 14:20:10 +01:00
|
|
|
command = '{} {}'.format(command, ' '.join(map(str, e.args)))
|
2017-03-06 11:10:25 +00:00
|
|
|
message = 'Command \'{}\' returned non-zero exit status {}\nOUTPUT:\n{}\n'
|
|
|
|
log_func(message.format(command, e.returncode, e.output))
|
|
|
|
elif isinstance(e, SyntaxError):
|
|
|
|
tb = get_traceback()
|
|
|
|
log_func(tb)
|
|
|
|
message = 'Syntax Error in {}, line {}, offset {}:'
|
|
|
|
log_func(message.format(e.filename, e.lineno, e.offset))
|
|
|
|
log_func('\t{}'.format(e.msg))
|
|
|
|
else:
|
|
|
|
tb = get_traceback()
|
|
|
|
log_func(tb)
|
|
|
|
log_func('{}({})'.format(e.__class__.__name__, e))
|
|
|
|
|
2017-09-27 10:13:39 +01:00
|
|
|
e.logged = True
|
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
|
|
|
|
class ErrorSignalHandler(logging.Handler):
|
|
|
|
"""
|
|
|
|
Emits signals for ERROR and WARNING level traces.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
def emit(self, record):
|
|
|
|
if record.levelno == logging.ERROR:
|
2018-03-07 11:32:19 +00:00
|
|
|
signal.send(signal.ERROR_LOGGED, self, record)
|
2017-03-06 11:10:25 +00:00
|
|
|
elif record.levelno == logging.WARNING:
|
2018-03-07 11:32:19 +00:00
|
|
|
signal.send(signal.WARNING_LOGGED, self, record)
|
2017-03-06 11:10:25 +00:00
|
|
|
|
|
|
|
|
2018-04-30 11:30:45 +01:00
|
|
|
class InitHandler(logging.handlers.BufferingHandler):
|
|
|
|
"""
|
|
|
|
Used to buffer early logging records before a log file is created.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, capacity):
|
|
|
|
super(InitHandler, self).__init__(capacity)
|
|
|
|
self.targets = []
|
|
|
|
|
2018-04-30 13:28:53 +01:00
|
|
|
def emit(self, record):
|
|
|
|
record.indent_level = _indent_level
|
|
|
|
super(InitHandler, self).emit(record)
|
2018-04-30 11:30:45 +01:00
|
|
|
|
|
|
|
def flush(self):
|
|
|
|
for target in self.targets:
|
|
|
|
self.flush_to_target(target)
|
|
|
|
self.buffer = []
|
|
|
|
|
2018-04-30 13:28:53 +01:00
|
|
|
def add_target(self, target):
|
|
|
|
if target not in self.targets:
|
|
|
|
self.targets.append(target)
|
|
|
|
|
2018-04-30 11:30:45 +01:00
|
|
|
def flush_to_target(self, target):
|
|
|
|
for record in self.buffer:
|
|
|
|
target.emit(record)
|
|
|
|
|
|
|
|
|
2017-03-06 11:10:25 +00:00
|
|
|
class LineFormatter(logging.Formatter):
|
|
|
|
"""
|
|
|
|
Logs each line of the message separately.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
def format(self, record):
|
|
|
|
record.message = record.getMessage()
|
|
|
|
if self.usesTime():
|
|
|
|
record.asctime = self.formatTime(record, self.datefmt)
|
|
|
|
|
2018-04-30 13:28:53 +01:00
|
|
|
indent_level = getattr(record, 'indent_level', _indent_level)
|
2018-07-03 12:33:24 +01:00
|
|
|
cur_indent = _indent_width * indent_level
|
2017-03-06 11:10:25 +00:00
|
|
|
d = record.__dict__
|
|
|
|
parts = []
|
|
|
|
for line in record.message.split('\n'):
|
2018-07-03 12:33:24 +01:00
|
|
|
line = ' ' * cur_indent + line
|
2017-03-06 11:10:25 +00:00
|
|
|
d.update({'message': line.strip('\r')})
|
|
|
|
parts.append(self._fmt % d)
|
|
|
|
|
|
|
|
return '\n'.join(parts)
|
|
|
|
|
|
|
|
|
|
|
|
class ColorFormatter(LineFormatter):
|
|
|
|
"""
|
|
|
|
Formats logging records with color and prepends record info
|
|
|
|
to each line of the message.
|
|
|
|
|
|
|
|
BLUE for DEBUG logging level
|
|
|
|
GREEN for INFO logging level
|
|
|
|
YELLOW for WARNING logging level
|
|
|
|
RED for ERROR logging level
|
|
|
|
BOLD RED for CRITICAL logging level
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, fmt=None, datefmt=None):
|
|
|
|
super(ColorFormatter, self).__init__(fmt, datefmt)
|
|
|
|
template_text = self._fmt.replace('%(message)s', RESET_COLOR + '%(message)s${color}')
|
|
|
|
template_text = '${color}' + template_text + RESET_COLOR
|
|
|
|
self.fmt_template = string.Template(template_text)
|
|
|
|
|
|
|
|
def format(self, record):
|
|
|
|
self._set_color(COLOR_MAP[record.levelno])
|
|
|
|
return super(ColorFormatter, self).format(record)
|
|
|
|
|
|
|
|
def _set_color(self, color):
|
|
|
|
self._fmt = self.fmt_template.substitute(color=color)
|
|
|
|
|
|
|
|
|
|
|
|
class BaseLogWriter(object):
|
|
|
|
|
|
|
|
def __init__(self, name, level=logging.DEBUG):
|
|
|
|
"""
|
|
|
|
File-like object class designed to be used for logging from streams
|
|
|
|
Each complete line (terminated by new line character) gets logged
|
|
|
|
at DEBUG level. In complete lines are buffered until the next new line.
|
|
|
|
|
|
|
|
:param name: The name of the logger that will be used.
|
|
|
|
|
|
|
|
"""
|
|
|
|
self.logger = logging.getLogger(name)
|
|
|
|
self.buffer = ''
|
|
|
|
if level == logging.DEBUG:
|
|
|
|
self.do_write = self.logger.debug
|
|
|
|
elif level == logging.INFO:
|
|
|
|
self.do_write = self.logger.info
|
|
|
|
elif level == logging.WARNING:
|
|
|
|
self.do_write = self.logger.warning
|
|
|
|
elif level == logging.ERROR:
|
|
|
|
self.do_write = self.logger.error
|
|
|
|
else:
|
|
|
|
raise Exception('Unknown logging level: {}'.format(level))
|
|
|
|
|
|
|
|
def flush(self):
|
|
|
|
# Defined to match the interface expected by pexpect.
|
|
|
|
return self
|
|
|
|
|
|
|
|
def close(self):
|
|
|
|
if self.buffer:
|
|
|
|
self.logger.debug(self.buffer)
|
|
|
|
self.buffer = ''
|
|
|
|
return self
|
|
|
|
|
|
|
|
def __del__(self):
|
|
|
|
# Ensure we don't lose bufferd output
|
|
|
|
self.close()
|
|
|
|
|
|
|
|
|
|
|
|
class LogWriter(BaseLogWriter):
|
|
|
|
|
|
|
|
def write(self, data):
|
|
|
|
data = data.replace('\r\n', '\n').replace('\r', '\n')
|
|
|
|
if '\n' in data:
|
|
|
|
parts = data.split('\n')
|
|
|
|
parts[0] = self.buffer + parts[0]
|
|
|
|
for part in parts[:-1]:
|
|
|
|
self.do_write(part)
|
|
|
|
self.buffer = parts[-1]
|
|
|
|
else:
|
|
|
|
self.buffer += data
|
|
|
|
return self
|
|
|
|
|
|
|
|
|
|
|
|
class LineLogWriter(BaseLogWriter):
|
|
|
|
|
|
|
|
def write(self, data):
|
|
|
|
self.do_write(data)
|
|
|
|
|
|
|
|
|
|
|
|
class StreamLogger(threading.Thread):
|
|
|
|
"""
|
|
|
|
Logs output from a stream in a thread.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, name, stream, level=logging.DEBUG, klass=LogWriter):
|
|
|
|
super(StreamLogger, self).__init__()
|
|
|
|
self.writer = klass(name, level)
|
|
|
|
self.stream = stream
|
|
|
|
self.daemon = True
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
line = self.stream.readline()
|
|
|
|
while line:
|
|
|
|
self.writer.write(line.rstrip('\n'))
|
|
|
|
line = self.stream.readline()
|
|
|
|
self.writer.close()
|