waf/waflib/Logs.py

380 lines
9.4 KiB
Python
Raw Normal View History

2011-09-10 11:13:51 +02:00
#!/usr/bin/env python
# encoding: utf-8
2018-01-01 20:53:49 +01:00
# Thomas Nagy, 2005-2018 (ita)
2011-09-10 11:13:51 +02:00
"""
logging, colors, terminal width and pretty-print
"""
import os, re, traceback, sys
2014-01-05 10:51:24 +01:00
from waflib import Utils, ansiterm
2011-09-10 11:13:51 +02:00
if not os.environ.get('NOSYNC', False):
# synchronized output is nearly mandatory to prevent garbled output
if sys.stdout.isatty() and id(sys.stdout) == id(sys.__stdout__):
sys.stdout = ansiterm.AnsiTerm(sys.stdout)
if sys.stderr.isatty() and id(sys.stderr) == id(sys.__stderr__):
sys.stderr = ansiterm.AnsiTerm(sys.stderr)
2012-04-04 00:54:46 +02:00
2014-01-14 02:31:08 +01:00
# import the logging module after since it holds a reference on sys.stderr
# in case someone uses the root logger
import logging
2011-09-10 11:13:51 +02:00
LOG_FORMAT = os.environ.get('WAF_LOG_FORMAT', '%(asctime)s %(c1)s%(zone)s%(c2)s %(message)s')
HOUR_FORMAT = os.environ.get('WAF_HOUR_FORMAT', '%H:%M:%S')
2011-09-10 11:13:51 +02:00
2016-06-25 12:50:04 +02:00
zones = []
"""
See :py:class:`waflib.Logs.log_filter`
"""
2011-09-10 11:13:51 +02:00
verbose = 0
2016-06-25 12:50:04 +02:00
"""
Global verbosity level, see :py:func:`waflib.Logs.debug` and :py:func:`waflib.Logs.error`
"""
2011-09-10 11:13:51 +02:00
colors_lst = {
'USE' : True,
'BOLD' :'\x1b[01;1m',
'RED' :'\x1b[01;31m',
'GREEN' :'\x1b[32m',
'YELLOW':'\x1b[33m',
'PINK' :'\x1b[35m',
'BLUE' :'\x1b[01;34m',
'CYAN' :'\x1b[36m',
2015-06-16 07:35:46 +02:00
'GREY' :'\x1b[37m',
2011-09-10 11:13:51 +02:00
'NORMAL':'\x1b[0m',
'cursor_on' :'\x1b[?25h',
'cursor_off' :'\x1b[?25l',
}
indicator = '\r\x1b[K%s%s%s'
2014-01-16 02:03:53 +01:00
try:
unicode
except NameError:
unicode = None
2014-01-05 10:51:24 +01:00
def enable_colors(use):
2016-06-25 12:50:04 +02:00
"""
If *1* is given, then the system will perform a few verifications
before enabling colors, such as checking whether the interpreter
is running in a terminal. A value of zero will disable colors,
and a value above *1* will force colors.
:param use: whether to enable colors or not
:type use: integer
"""
2014-01-05 10:51:24 +01:00
if use == 1:
if not (sys.stderr.isatty() or sys.stdout.isatty()):
use = 0
if Utils.is_win32 and os.name != 'java':
2014-01-05 10:51:24 +01:00
term = os.environ.get('TERM', '') # has ansiterm
else:
term = os.environ.get('TERM', 'dumb')
if term in ('dumb', 'emacs'):
use = 0
if use >= 1:
os.environ['TERM'] = 'vt100'
colors_lst['USE'] = use
2011-09-10 11:13:51 +02:00
# If console packages are available, replace the dummy function with a real
# implementation
try:
2014-01-05 11:23:49 +01:00
get_term_cols = ansiterm.get_term_cols
except AttributeError:
def get_term_cols():
return 80
2011-09-10 11:13:51 +02:00
get_term_cols.__doc__ = """
2016-06-25 12:50:04 +02:00
Returns the console width in characters.
2011-09-10 11:13:51 +02:00
:return: the number of characters per line
:rtype: int
"""
def get_color(cl):
2016-06-25 12:50:04 +02:00
"""
Returns the ansi sequence corresponding to the given color name.
An empty string is returned when coloring is globally disabled.
:param cl: color name in capital letters
:type cl: string
"""
2016-06-25 02:38:26 +02:00
if colors_lst['USE']:
return colors_lst.get(cl, '')
return ''
2011-09-10 11:13:51 +02:00
class color_dict(object):
"""attribute-based color access, eg: colors.PINK"""
def __getattr__(self, a):
return get_color(a)
def __call__(self, a):
return get_color(a)
colors = color_dict()
re_log = re.compile(r'(\w+): (.*)', re.M)
class log_filter(logging.Filter):
"""
2016-06-25 12:50:04 +02:00
Waf logs are of the form 'name: message', and can be filtered by 'waf --zones=name'.
2011-09-10 11:13:51 +02:00
For example, the following::
from waflib import Logs
Logs.debug('test: here is a message')
Will be displayed only when executing::
$ waf --zones=test
"""
def __init__(self, name=''):
logging.Filter.__init__(self, name)
2011-09-10 11:13:51 +02:00
def filter(self, rec):
"""
2016-06-25 12:50:04 +02:00
Filters log records by zone and by logging level
2011-09-10 11:13:51 +02:00
2016-06-25 12:50:04 +02:00
:param rec: log entry
2011-09-10 11:13:51 +02:00
"""
rec.zone = rec.module
if rec.levelno >= logging.INFO:
return True
m = re_log.match(rec.msg)
if m:
rec.zone = m.group(1)
rec.msg = m.group(2)
if zones:
return getattr(rec, 'zone', '') in zones or '*' in zones
elif not verbose > 2:
return False
return True
2014-01-14 02:31:08 +01:00
class log_handler(logging.StreamHandler):
"""Dispatches messages to stderr/stdout depending on the severity level"""
def emit(self, record):
2016-06-25 12:50:04 +02:00
"""
Delegates the functionality to :py:meth:`waflib.Log.log_handler.emit_override`
"""
2014-01-14 02:31:08 +01:00
# default implementation
try:
try:
self.stream = record.stream
except AttributeError:
if record.levelno >= logging.WARNING:
record.stream = self.stream = sys.stderr
else:
record.stream = self.stream = sys.stdout
2014-01-14 02:31:08 +01:00
self.emit_override(record)
self.flush()
except (KeyboardInterrupt, SystemExit):
raise
2014-04-20 02:29:27 +02:00
except: # from the python library -_-
2014-01-14 02:31:08 +01:00
self.handleError(record)
def emit_override(self, record, **kw):
2016-06-25 12:50:04 +02:00
"""
Writes the log record to the desired stream (stderr/stdout)
"""
2014-01-14 02:31:08 +01:00
self.terminator = getattr(record, 'terminator', '\n')
stream = self.stream
if unicode:
2014-01-14 02:31:08 +01:00
# python2
msg = self.formatter.format(record)
fs = '%s' + self.terminator
try:
if (isinstance(msg, unicode) and getattr(stream, 'encoding', None)):
fs = fs.decode(stream.encoding)
try:
stream.write(fs % msg)
except UnicodeEncodeError:
stream.write((fs % msg).encode(stream.encoding))
else:
stream.write(fs % msg)
except UnicodeError:
stream.write((fs % msg).encode('utf-8'))
2014-01-14 02:31:08 +01:00
else:
logging.StreamHandler.emit(self, record)
2011-09-10 11:13:51 +02:00
class formatter(logging.Formatter):
"""Simple log formatter which handles colors"""
def __init__(self):
logging.Formatter.__init__(self, LOG_FORMAT, HOUR_FORMAT)
def format(self, rec):
2016-06-25 12:50:04 +02:00
"""
Formats records and adds colors as needed. The records do not get
a leading hour format if the logging level is above *INFO*.
"""
2014-01-14 02:31:08 +01:00
try:
msg = rec.msg.decode('utf-8')
except Exception:
msg = rec.msg
use = colors_lst['USE']
if (use == 1 and rec.stream.isatty()) or use == 2:
2014-01-14 02:31:08 +01:00
c1 = getattr(rec, 'c1', None)
2014-01-16 02:03:53 +01:00
if c1 is None:
2014-01-14 02:31:08 +01:00
c1 = ''
if rec.levelno >= logging.ERROR:
c1 = colors.RED
elif rec.levelno >= logging.WARNING:
c1 = colors.YELLOW
elif rec.levelno >= logging.INFO:
c1 = colors.GREEN
c2 = getattr(rec, 'c2', colors.NORMAL)
2014-07-25 18:15:23 +02:00
msg = '%s%s%s' % (c1, msg, c2)
2014-01-14 02:31:08 +01:00
else:
# remove single \r that make long lines in text files
# and other terminal commands
msg = re.sub(r'\r(?!\n)|\x1B\[(K|.*?(m|h|l))', '', msg)
2014-01-14 02:31:08 +01:00
2016-05-28 15:11:58 +02:00
if rec.levelno >= logging.INFO:
# the goal of this is to format without the leading "Logs, hour" prefix
if rec.args:
return msg % rec.args
2014-01-14 02:31:08 +01:00
return msg
rec.msg = msg
rec.c1 = colors.PINK
rec.c2 = colors.NORMAL
2011-09-10 11:13:51 +02:00
return logging.Formatter.format(self, rec)
log = None
"""global logger for Logs.debug, Logs.error, etc"""
def debug(*k, **kw):
"""
2016-06-25 12:50:04 +02:00
Wraps logging.debug and discards messages if the verbosity level :py:attr:`waflib.Logs.verbose` 0
2011-09-10 11:13:51 +02:00
"""
if verbose:
k = list(k)
k[0] = k[0].replace('\n', ' ')
log.debug(*k, **kw)
def error(*k, **kw):
"""
2016-06-25 12:50:04 +02:00
Wrap logging.errors, adds the stack trace when the verbosity level :py:attr:`waflib.Logs.verbose` 2
2011-09-10 11:13:51 +02:00
"""
log.error(*k, **kw)
if verbose > 2:
st = traceback.extract_stack()
if st:
st = st[:-1]
buf = []
for filename, lineno, name, line in st:
2016-06-25 12:50:04 +02:00
buf.append(' File %r, line %d, in %s' % (filename, lineno, name))
2011-09-10 11:13:51 +02:00
if line:
buf.append(' %s' % line.strip())
if buf:
log.error('\n'.join(buf))
2011-09-10 11:13:51 +02:00
def warn(*k, **kw):
"""
2019-01-16 06:51:14 +01:00
Wraps logging.warning
2011-09-10 11:13:51 +02:00
"""
2019-01-16 06:51:14 +01:00
log.warning(*k, **kw)
2011-09-10 11:13:51 +02:00
def info(*k, **kw):
"""
2016-06-25 12:50:04 +02:00
Wraps logging.info
2011-09-10 11:13:51 +02:00
"""
log.info(*k, **kw)
def init_log():
"""
2016-06-25 12:50:04 +02:00
Initializes the logger :py:attr:`waflib.Logs.log`
2011-09-10 11:13:51 +02:00
"""
global log
log = logging.getLogger('waflib')
log.handlers = []
log.filters = []
2014-01-14 02:31:08 +01:00
hdlr = log_handler()
2011-09-10 11:13:51 +02:00
hdlr.setFormatter(formatter())
log.addHandler(hdlr)
log.addFilter(log_filter())
log.setLevel(logging.DEBUG)
def make_logger(path, name):
"""
2016-06-25 12:50:04 +02:00
Creates a simple logger, which is often used to redirect the context command output::
2011-09-10 11:13:51 +02:00
from waflib import Logs
bld.logger = Logs.make_logger('test.log', 'build')
bld.check(header_name='sadlib.h', features='cxx cprogram', mandatory=False)
2014-04-01 23:06:10 +02:00
# have the file closed immediately
Logs.free_logger(bld.logger)
# stop logging
2011-09-10 11:13:51 +02:00
bld.logger = None
2014-04-01 23:06:10 +02:00
The method finalize() of the command will try to free the logger, if any
2011-09-10 11:13:51 +02:00
:param path: file name to write the log output to
:type path: string
:param name: logger name (loggers are reused)
:type name: string
"""
logger = logging.getLogger(name)
if sys.hexversion > 0x3000000:
encoding = sys.stdout.encoding
else:
encoding = None
hdlr = logging.FileHandler(path, 'w', encoding=encoding)
2011-09-10 11:13:51 +02:00
formatter = logging.Formatter('%(message)s')
hdlr.setFormatter(formatter)
logger.addHandler(hdlr)
logger.setLevel(logging.DEBUG)
return logger
2014-04-01 23:06:10 +02:00
def make_mem_logger(name, to_log, size=8192):
2011-09-10 11:13:51 +02:00
"""
2016-06-25 12:50:04 +02:00
Creates a memory logger to avoid writing concurrently to the main logger
2011-09-10 11:13:51 +02:00
"""
from logging.handlers import MemoryHandler
logger = logging.getLogger(name)
hdlr = MemoryHandler(size, target=to_log)
formatter = logging.Formatter('%(message)s')
hdlr.setFormatter(formatter)
logger.addHandler(hdlr)
logger.memhandler = hdlr
logger.setLevel(logging.DEBUG)
return logger
2014-04-01 23:06:10 +02:00
def free_logger(logger):
"""
2016-06-25 12:50:04 +02:00
Frees the resources held by the loggers created through make_logger or make_mem_logger.
2014-04-01 23:06:10 +02:00
This is used for file cleanup and for handler removal (logger objects are re-used).
"""
try:
for x in logger.handlers:
x.close()
logger.removeHandler(x)
except Exception:
2014-04-01 23:06:10 +02:00
pass
def pprint(col, msg, label='', sep='\n'):
2011-09-10 11:13:51 +02:00
"""
2016-06-25 12:50:04 +02:00
Prints messages in color immediately on stderr::
2011-09-10 11:13:51 +02:00
from waflib import Logs
Logs.pprint('RED', 'Something bad just happened')
:param col: color name to use in :py:const:`Logs.colors_lst`
:type col: string
2014-04-01 23:06:10 +02:00
:param msg: message to display
:type msg: string or a value that can be printed by %s
2011-09-10 11:13:51 +02:00
:param label: a message to add after the colored output
:type label: string
:param sep: a string to append at the end (line separator)
:type sep: string
"""
info('%s%s%s %s', colors(col), msg, colors.NORMAL, label, extra={'terminator':sep})
2011-09-10 11:13:51 +02:00