logger.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. import enum
  2. import logging
  3. import logging.handlers
  4. import os
  5. import sys
  6. def to_bool(arg):
  7. '''
  8. Return a boolean value based on `arg`.
  9. '''
  10. if arg is None or isinstance(arg, bool):
  11. return arg
  12. if isinstance(arg, str):
  13. arg = arg.lower()
  14. if arg in ('yes', 'on', '1', 'true', 1):
  15. return True
  16. return False
  17. def interactive_console():
  18. '''
  19. Return whether the current console is "interactive". Meaning: Capable of
  20. user input and not just something like a cron job.
  21. '''
  22. return sys.stderr.isatty() and os.environ.get('TERM') != 'dumb'
  23. def should_do_markup(no_color, configs):
  24. '''
  25. Given the value of the command-line no-color argument, and a dict of configuration filename to
  26. corresponding parsed configuration, determine if we should enable color marking up.
  27. '''
  28. if no_color:
  29. return False
  30. if any(config.get('color', True) is False for config in configs.values()):
  31. return False
  32. if os.environ.get('NO_COLOR', None):
  33. return False
  34. py_colors = os.environ.get('PY_COLORS', None)
  35. if py_colors is not None:
  36. return to_bool(py_colors)
  37. return interactive_console()
  38. class Multi_stream_handler(logging.Handler):
  39. '''
  40. A logging handler that dispatches each log record to one of multiple stream handlers depending
  41. on the record's log level.
  42. '''
  43. def __init__(self, log_level_to_stream_handler):
  44. super(Multi_stream_handler, self).__init__()
  45. self.log_level_to_handler = log_level_to_stream_handler
  46. self.handlers = set(self.log_level_to_handler.values())
  47. def flush(self): # pragma: no cover
  48. super(Multi_stream_handler, self).flush()
  49. for handler in self.handlers:
  50. handler.flush()
  51. def emit(self, record):
  52. '''
  53. Dispatch the log record to the appropriate stream handler for the record's log level.
  54. '''
  55. self.log_level_to_handler[record.levelno].emit(record)
  56. def setFormatter(self, formatter): # pragma: no cover
  57. super(Multi_stream_handler, self).setFormatter(formatter)
  58. for handler in self.handlers:
  59. handler.setFormatter(formatter)
  60. def setLevel(self, level): # pragma: no cover
  61. super(Multi_stream_handler, self).setLevel(level)
  62. for handler in self.handlers:
  63. handler.setLevel(level)
  64. class Console_no_color_formatter(logging.Formatter):
  65. def __init__(self, *args, **kwargs): # pragma: no cover
  66. super(Console_no_color_formatter, self).__init__(
  67. '{prefix}{message}', style='{', defaults={'prefix': ''}, *args, **kwargs
  68. )
  69. class Color(enum.Enum):
  70. RESET = 0
  71. RED = 31
  72. GREEN = 32
  73. YELLOW = 33
  74. MAGENTA = 35
  75. CYAN = 36
  76. class Console_color_formatter(logging.Formatter):
  77. def __init__(self, *args, **kwargs):
  78. super(Console_color_formatter, self).__init__(
  79. '{prefix}{message}', style='{', defaults={'prefix': ''}, *args, **kwargs
  80. )
  81. def format(self, record):
  82. add_custom_log_levels()
  83. color = (
  84. {
  85. logging.CRITICAL: Color.RED,
  86. logging.ERROR: Color.RED,
  87. logging.WARN: Color.YELLOW,
  88. logging.ANSWER: Color.MAGENTA,
  89. logging.INFO: Color.GREEN,
  90. logging.DEBUG: Color.CYAN,
  91. }
  92. .get(record.levelno)
  93. .value
  94. )
  95. return color_text(color, super(Console_color_formatter, self).format(record))
  96. def ansi_escape_code(color): # pragma: no cover
  97. '''
  98. Given a color value, produce the corresponding ANSI escape code.
  99. '''
  100. return f'\x1b[{color}m'
  101. def color_text(color, message):
  102. '''
  103. Given a color value and a message, return the message colored with ANSI escape codes.
  104. '''
  105. if not color:
  106. return message
  107. return f'{ansi_escape_code(color)}{message}{ansi_escape_code(Color.RESET.value)}'
  108. def add_logging_level(level_name, level_number):
  109. '''
  110. Globally add a custom logging level based on the given (all uppercase) level name and number.
  111. Do this idempotently.
  112. Inspired by https://stackoverflow.com/questions/2183233/how-to-add-a-custom-loglevel-to-pythons-logging-facility/35804945#35804945
  113. '''
  114. method_name = level_name.lower()
  115. if not hasattr(logging, level_name):
  116. logging.addLevelName(level_number, level_name)
  117. setattr(logging, level_name, level_number)
  118. if not hasattr(logging, method_name):
  119. def log_for_level(self, message, *args, **kwargs): # pragma: no cover
  120. if self.isEnabledFor(level_number):
  121. self._log(level_number, message, args, **kwargs)
  122. setattr(logging.getLoggerClass(), method_name, log_for_level)
  123. if not hasattr(logging.getLoggerClass(), method_name):
  124. def log_to_root(message, *args, **kwargs): # pragma: no cover
  125. logging.log(level_number, message, *args, **kwargs)
  126. setattr(logging, method_name, log_to_root)
  127. ANSWER = logging.WARN - 5
  128. DISABLED = logging.CRITICAL + 10
  129. def add_custom_log_levels(): # pragma: no cover
  130. '''
  131. Add a custom log level between WARN and INFO for user-requested answers.
  132. '''
  133. add_logging_level('ANSWER', ANSWER)
  134. add_logging_level('DISABLED', DISABLED)
  135. def get_log_prefix():
  136. '''
  137. Return the current log prefix from the defaults for the formatter on the first logging handler,
  138. set by set_log_prefix(). Return None if no such prefix exists.
  139. '''
  140. try:
  141. return next(
  142. handler.formatter._style._defaults.get('prefix').rstrip().rstrip(':')
  143. for handler in logging.getLogger().handlers
  144. )
  145. except (StopIteration, AttributeError):
  146. return None
  147. def set_log_prefix(prefix):
  148. '''
  149. Given a log prefix as a string, set it into the defaults for the formatters on all logging
  150. handlers.
  151. '''
  152. for handler in logging.getLogger().handlers:
  153. try:
  154. handler.formatter._style._defaults = {'prefix': f'{prefix}: ' if prefix else ''}
  155. except AttributeError:
  156. pass
  157. class Log_prefix:
  158. '''
  159. A Python context manager for setting a log prefix so that it shows up in every subsequent
  160. logging message for the duration of the context manager. For this to work, it relies on each
  161. logging formatter to be initialized with "{prefix}" somewhere in its logging format.
  162. Example use as a context manager:
  163. with borgmatic.logger.Log_prefix('myprefix'):
  164. do_something_that_logs()
  165. For the scope of that "with" statement, any logs created are prefixed with "myprefix: ".
  166. Afterwards, the prefix gets restored to whatever it was prior to the context manager.
  167. '''
  168. def __init__(self, prefix):
  169. '''
  170. Given the desired log prefix, save it for use below. Set prefix to None to disable any
  171. prefix from getting logged.
  172. '''
  173. self.prefix = prefix
  174. self.original_prefix = None
  175. def __enter__(self):
  176. '''
  177. Set the prefix onto the formatter defaults for every logging handler so that the prefix ends
  178. up in every log message. But first, save off any original prefix so that it can be restored
  179. below.
  180. '''
  181. self.original_prefix = get_log_prefix()
  182. set_log_prefix(self.prefix)
  183. def __exit__(self, exception, value, traceback):
  184. '''
  185. Restore any original prefix.
  186. '''
  187. set_log_prefix(self.original_prefix)
  188. def configure_logging(
  189. console_log_level,
  190. syslog_log_level=None,
  191. log_file_log_level=None,
  192. monitoring_log_level=None,
  193. log_file=None,
  194. log_file_format=None,
  195. color_enabled=True,
  196. ):
  197. '''
  198. Configure logging to go to both the console and (syslog or log file). Use the given log levels,
  199. respectively. If color is enabled, set up log formatting accordingly.
  200. Raise FileNotFoundError or PermissionError if the log file could not be opened for writing.
  201. '''
  202. add_custom_log_levels()
  203. if syslog_log_level is None:
  204. syslog_log_level = logging.DISABLED
  205. if log_file_log_level is None:
  206. log_file_log_level = console_log_level
  207. if monitoring_log_level is None:
  208. monitoring_log_level = console_log_level
  209. # Log certain log levels to console stderr and others to stdout. This supports use cases like
  210. # grepping (non-error) output.
  211. console_disabled = logging.NullHandler()
  212. console_error_handler = logging.StreamHandler(sys.stderr)
  213. console_standard_handler = logging.StreamHandler(sys.stdout)
  214. console_handler = Multi_stream_handler(
  215. {
  216. logging.DISABLED: console_disabled,
  217. logging.CRITICAL: console_error_handler,
  218. logging.ERROR: console_error_handler,
  219. logging.WARN: console_error_handler,
  220. logging.ANSWER: console_standard_handler,
  221. logging.INFO: console_standard_handler,
  222. logging.DEBUG: console_standard_handler,
  223. }
  224. )
  225. if color_enabled:
  226. console_handler.setFormatter(Console_color_formatter())
  227. else:
  228. console_handler.setFormatter(Console_no_color_formatter())
  229. console_handler.setLevel(console_log_level)
  230. handlers = [console_handler]
  231. if syslog_log_level != logging.DISABLED:
  232. syslog_path = None
  233. if os.path.exists('/dev/log'):
  234. syslog_path = '/dev/log'
  235. elif os.path.exists('/var/run/syslog'):
  236. syslog_path = '/var/run/syslog'
  237. elif os.path.exists('/var/run/log'):
  238. syslog_path = '/var/run/log'
  239. if syslog_path:
  240. syslog_handler = logging.handlers.SysLogHandler(address=syslog_path)
  241. syslog_handler.setFormatter(
  242. logging.Formatter(
  243. 'borgmatic: {levelname} {prefix}{message}', # noqa: FS003
  244. style='{',
  245. defaults={'prefix': ''},
  246. )
  247. )
  248. syslog_handler.setLevel(syslog_log_level)
  249. handlers.append(syslog_handler)
  250. if log_file and log_file_log_level != logging.DISABLED:
  251. file_handler = logging.handlers.WatchedFileHandler(log_file)
  252. file_handler.setFormatter(
  253. logging.Formatter(
  254. log_file_format or '[{asctime}] {levelname}: {prefix}{message}', # noqa: FS003
  255. style='{',
  256. defaults={'prefix': ''},
  257. )
  258. )
  259. file_handler.setLevel(log_file_log_level)
  260. handlers.append(file_handler)
  261. logging.basicConfig(
  262. level=min(handler.level for handler in handlers),
  263. handlers=handlers,
  264. )