logger.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. import logging
  2. import logging.handlers
  3. import os
  4. import sys
  5. import colorama
  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 colorama marking up.
  27. '''
  28. if no_color:
  29. return False
  30. if any(config.get('output', {}).get('color') is False for config in configs.values()):
  31. return False
  32. py_colors = os.environ.get('PY_COLORS', None)
  33. if py_colors is not None:
  34. return to_bool(py_colors)
  35. return interactive_console()
  36. class Multi_stream_handler(logging.Handler):
  37. '''
  38. A logging handler that dispatches each log record to one of multiple stream handlers depending
  39. on the record's log level.
  40. '''
  41. def __init__(self, log_level_to_stream_handler):
  42. super(Multi_stream_handler, self).__init__()
  43. self.log_level_to_handler = log_level_to_stream_handler
  44. self.handlers = set(self.log_level_to_handler.values())
  45. def flush(self): # pragma: no cover
  46. super(Multi_stream_handler, self).flush()
  47. for handler in self.handlers:
  48. handler.flush()
  49. def emit(self, record):
  50. '''
  51. Dispatch the log record to the appropriate stream handler for the record's log level.
  52. '''
  53. self.log_level_to_handler[record.levelno].emit(record)
  54. def setFormatter(self, formatter): # pragma: no cover
  55. super(Multi_stream_handler, self).setFormatter(formatter)
  56. for handler in self.handlers:
  57. handler.setFormatter(formatter)
  58. def setLevel(self, level): # pragma: no cover
  59. super(Multi_stream_handler, self).setLevel(level)
  60. for handler in self.handlers:
  61. handler.setLevel(level)
  62. class Console_color_formatter(logging.Formatter):
  63. def format(self, record):
  64. add_custom_log_levels()
  65. color = {
  66. logging.CRITICAL: colorama.Fore.RED,
  67. logging.ERROR: colorama.Fore.RED,
  68. logging.WARN: colorama.Fore.YELLOW,
  69. logging.ANSWER: colorama.Fore.MAGENTA,
  70. logging.INFO: colorama.Fore.GREEN,
  71. logging.DEBUG: colorama.Fore.CYAN,
  72. }.get(record.levelno)
  73. return color_text(color, record.msg)
  74. def color_text(color, message):
  75. '''
  76. Give colored text.
  77. '''
  78. if not color:
  79. return message
  80. return f'{color}{message}{colorama.Style.RESET_ALL}'
  81. def add_logging_level(level_name, level_number):
  82. '''
  83. Globally add a custom logging level based on the given (all uppercase) level name and number.
  84. Do this idempotently.
  85. Inspired by https://stackoverflow.com/questions/2183233/how-to-add-a-custom-loglevel-to-pythons-logging-facility/35804945#35804945
  86. '''
  87. method_name = level_name.lower()
  88. if not hasattr(logging, level_name):
  89. logging.addLevelName(level_number, level_name)
  90. setattr(logging, level_name, level_number)
  91. if not hasattr(logging, method_name):
  92. def log_for_level(self, message, *args, **kwargs): # pragma: no cover
  93. if self.isEnabledFor(level_number):
  94. self._log(level_number, message, args, **kwargs)
  95. setattr(logging.getLoggerClass(), method_name, log_for_level)
  96. if not hasattr(logging.getLoggerClass(), method_name):
  97. def log_to_root(message, *args, **kwargs): # pragma: no cover
  98. logging.log(level_number, message, *args, **kwargs)
  99. setattr(logging, method_name, log_to_root)
  100. ANSWER = logging.WARN - 5
  101. DISABLED = logging.CRITICAL + 10
  102. def add_custom_log_levels(): # pragma: no cover
  103. '''
  104. Add a custom log level between WARN and INFO for user-requested answers.
  105. '''
  106. add_logging_level('ANSWER', ANSWER)
  107. add_logging_level('DISABLED', DISABLED)
  108. def configure_logging(
  109. console_log_level,
  110. syslog_log_level=None,
  111. log_file_log_level=None,
  112. monitoring_log_level=None,
  113. log_file=None,
  114. log_file_format=None,
  115. ):
  116. '''
  117. Configure logging to go to both the console and (syslog or log file). Use the given log levels,
  118. respectively.
  119. Raise FileNotFoundError or PermissionError if the log file could not be opened for writing.
  120. '''
  121. if syslog_log_level is None:
  122. syslog_log_level = console_log_level
  123. if log_file_log_level is None:
  124. log_file_log_level = console_log_level
  125. if monitoring_log_level is None:
  126. monitoring_log_level = console_log_level
  127. add_custom_log_levels()
  128. # Log certain log levels to console stderr and others to stdout. This supports use cases like
  129. # grepping (non-error) output.
  130. console_disabled = logging.NullHandler()
  131. console_error_handler = logging.StreamHandler(sys.stderr)
  132. console_standard_handler = logging.StreamHandler(sys.stdout)
  133. console_handler = Multi_stream_handler(
  134. {
  135. logging.DISABLED: console_disabled,
  136. logging.CRITICAL: console_error_handler,
  137. logging.ERROR: console_error_handler,
  138. logging.WARN: console_error_handler,
  139. logging.ANSWER: console_standard_handler,
  140. logging.INFO: console_standard_handler,
  141. logging.DEBUG: console_standard_handler,
  142. }
  143. )
  144. console_handler.setFormatter(Console_color_formatter())
  145. console_handler.setLevel(console_log_level)
  146. syslog_path = None
  147. if log_file is None and syslog_log_level != logging.DISABLED:
  148. if os.path.exists('/dev/log'):
  149. syslog_path = '/dev/log'
  150. elif os.path.exists('/var/run/syslog'):
  151. syslog_path = '/var/run/syslog'
  152. elif os.path.exists('/var/run/log'):
  153. syslog_path = '/var/run/log'
  154. if syslog_path and not interactive_console():
  155. syslog_handler = logging.handlers.SysLogHandler(address=syslog_path)
  156. syslog_handler.setFormatter(
  157. logging.Formatter('borgmatic: {levelname} {message}', style='{') # noqa: FS003
  158. )
  159. syslog_handler.setLevel(syslog_log_level)
  160. handlers = (console_handler, syslog_handler)
  161. elif log_file and log_file_log_level != logging.DISABLED:
  162. file_handler = logging.handlers.WatchedFileHandler(log_file)
  163. file_handler.setFormatter(
  164. logging.Formatter(
  165. log_file_format or '[{asctime}] {levelname}: {message}', style='{' # noqa: FS003
  166. )
  167. )
  168. file_handler.setLevel(log_file_log_level)
  169. handlers = (console_handler, file_handler)
  170. else:
  171. handlers = (console_handler,)
  172. logging.basicConfig(
  173. level=min(console_log_level, syslog_log_level, log_file_log_level, monitoring_log_level),
  174. handlers=handlers,
  175. )