forked from pyinfra-dev/pyinfra
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlog.py
66 lines (48 loc) · 1.92 KB
/
log.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
from __future__ import division, print_function, unicode_literals
import logging
import click
import six
from pyinfra import logger
STDOUT_LOG_LEVELS = (logging.DEBUG, logging.INFO)
STDERR_LOG_LEVELS = (logging.WARNING, logging.ERROR, logging.CRITICAL)
class LogHandler(logging.Handler):
def emit(self, record):
try:
message = self.format(record)
click.echo(message, err=True)
except Exception:
self.handleError(record)
class LogFormatter(logging.Formatter):
level_to_format = {
logging.DEBUG: lambda s: click.style(s, 'green'),
logging.WARNING: lambda s: click.style(s, 'yellow'),
logging.ERROR: lambda s: click.style(s, 'red'),
logging.CRITICAL: lambda s: click.style(s, 'red', bold=True),
}
def format(self, record):
message = record.msg
if record.args:
message = record.msg % record.args
# Add path/module info for debug
if record.levelno is logging.DEBUG:
path_start = record.pathname.rfind('pyinfra')
if path_start:
pyinfra_path = record.pathname[path_start:-3] # -3 removes `.py`
module_name = pyinfra_path.replace('/', '.')
message = '[{0}] {1}'.format(module_name, message)
# We only handle strings here
if isinstance(message, six.string_types):
if '-->' not in message:
message = ' {0}'.format(message)
if record.levelno in self.level_to_format:
message = self.level_to_format[record.levelno](message)
return message
# If not a string, pass to standard Formatter
else:
return super(LogFormatter, self).format(record)
def setup_logging(log_level):
logger.setLevel(log_level)
handler = LogHandler()
formatter = LogFormatter()
handler.setFormatter(formatter)
logger.addHandler(handler)