logger.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410
  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. return arg in {'yes', 'on', '1', 'true', 1}
  15. def interactive_console():
  16. '''
  17. Return whether the current console is "interactive". Meaning: Capable of
  18. user input and not just something like a cron job.
  19. '''
  20. return sys.stderr.isatty() and os.environ.get('TERM') != 'dumb'
  21. def should_do_markup(configs, json_enabled):
  22. '''
  23. Given a dict of configuration filename to corresponding parsed configuration (which already have
  24. any command-line overrides applied) and whether json is enabled, determine if we should enable
  25. color marking up.
  26. '''
  27. if json_enabled:
  28. return False
  29. if any(config.get('color', True) is False for config in configs.values()):
  30. return False
  31. if os.environ.get('NO_COLOR', None):
  32. return False
  33. py_colors = os.environ.get('PY_COLORS', None)
  34. if py_colors is not None:
  35. return to_bool(py_colors)
  36. return interactive_console()
  37. class Multi_stream_handler(logging.Handler):
  38. '''
  39. A logging handler that dispatches each log record to one of multiple stream handlers depending
  40. on the record's log level.
  41. '''
  42. def __init__(self, log_level_to_stream_handler):
  43. super().__init__()
  44. self.log_level_to_handler = log_level_to_stream_handler
  45. self.handlers = set(self.log_level_to_handler.values())
  46. def flush(self): # pragma: no cover
  47. super().flush()
  48. for handler in self.handlers:
  49. handler.flush()
  50. def emit(self, record):
  51. '''
  52. Dispatch the log record to the appropriate stream handler for the record's log level.
  53. '''
  54. self.log_level_to_handler[record.levelno].emit(record)
  55. def setFormatter(self, formatter): # pragma: no cover # noqa: N802
  56. super().setFormatter(formatter)
  57. for handler in self.handlers:
  58. handler.setFormatter(formatter)
  59. def setLevel(self, level): # pragma: no cover # noqa: N802
  60. super().setLevel(level)
  61. for handler in self.handlers:
  62. handler.setLevel(level)
  63. class Log_prefix_formatter(logging.Formatter):
  64. def __init__(self, fmt='{prefix}{message}', *args, style='{', **kwargs):
  65. self.prefix = None
  66. super().__init__(*args, fmt=fmt, style=style, **kwargs)
  67. def format(self, record): # pragma: no cover
  68. record.prefix = f'{self.prefix}: ' if self.prefix else ''
  69. return super().format(record)
  70. class Color(enum.Enum):
  71. RESET = 0
  72. RED = 31
  73. GREEN = 32
  74. YELLOW = 33
  75. MAGENTA = 35
  76. CYAN = 36
  77. class Console_color_formatter(logging.Formatter):
  78. def __init__(self, *args, **kwargs):
  79. self.prefix = None
  80. super().__init__(
  81. '{prefix}{message}',
  82. *args,
  83. style='{',
  84. **kwargs,
  85. )
  86. def format(self, record):
  87. add_custom_log_levels()
  88. color = {
  89. logging.CRITICAL: Color.RED,
  90. logging.ERROR: Color.RED,
  91. logging.WARNING: Color.YELLOW,
  92. logging.ANSWER: Color.MAGENTA,
  93. logging.INFO: Color.GREEN,
  94. logging.DEBUG: Color.CYAN,
  95. }.get(record.levelno).value
  96. record.prefix = f'{self.prefix}: ' if self.prefix else ''
  97. return color_text(color, super().format(record))
  98. def ansi_escape_code(color): # pragma: no cover
  99. '''
  100. Given a color value, produce the corresponding ANSI escape code.
  101. '''
  102. return f'\x1b[{color}m'
  103. def color_text(color, message):
  104. '''
  105. Given a color value and a message, return the message colored with ANSI escape codes.
  106. '''
  107. if not color:
  108. return message
  109. return f'{ansi_escape_code(color)}{message}{ansi_escape_code(Color.RESET.value)}'
  110. def add_logging_level(level_name, level_number):
  111. '''
  112. Globally add a custom logging level based on the given (all uppercase) level name and number.
  113. Do this idempotently.
  114. Inspired by https://stackoverflow.com/questions/2183233/how-to-add-a-custom-loglevel-to-pythons-logging-facility/35804945#35804945
  115. '''
  116. method_name = level_name.lower()
  117. if not hasattr(logging, level_name):
  118. logging.addLevelName(level_number, level_name)
  119. setattr(logging, level_name, level_number)
  120. if not hasattr(logging, method_name):
  121. def log_for_level(self, message, *args, **kwargs): # pragma: no cover
  122. if self.isEnabledFor(level_number):
  123. self._log(level_number, message, args, **kwargs)
  124. setattr(logging.getLoggerClass(), method_name, log_for_level)
  125. if not hasattr(logging.getLoggerClass(), method_name):
  126. def log_to_root(message, *args, **kwargs): # pragma: no cover
  127. logging.log(level_number, message, *args, **kwargs) # noqa: LOG015
  128. setattr(logging, method_name, log_to_root)
  129. ANSWER = logging.WARNING - 5
  130. DISABLED = logging.CRITICAL + 10
  131. def add_custom_log_levels(): # pragma: no cover
  132. '''
  133. Add a custom log level between WARNING and INFO for user-requested answers.
  134. '''
  135. add_logging_level('ANSWER', ANSWER)
  136. add_logging_level('DISABLED', DISABLED)
  137. def get_log_prefix():
  138. '''
  139. Return the current log prefix set by set_log_prefix(). Return None if no such prefix exists.
  140. It would be a whole lot easier to use logger.Formatter(defaults=...) instead, but that argument
  141. doesn't exist until Python 3.10+.
  142. '''
  143. try:
  144. formatter = next(
  145. handler.formatter
  146. for handler in logging.getLogger().handlers
  147. if handler.formatter
  148. if hasattr(handler.formatter, 'prefix')
  149. )
  150. except StopIteration:
  151. return None
  152. return formatter.prefix
  153. def set_log_prefix(prefix):
  154. '''
  155. Given a log prefix as a string, set it into the each handler's formatter so that it can inject
  156. the prefix into each logged record.
  157. '''
  158. for handler in logging.getLogger().handlers:
  159. if handler.formatter and hasattr(handler.formatter, 'prefix'):
  160. handler.formatter.prefix = prefix
  161. class Log_prefix:
  162. '''
  163. A Python context manager for setting a log prefix so that it shows up in every subsequent
  164. logging message for the duration of the context manager. For this to work, it relies on each
  165. logging formatter to be initialized with "{prefix}" somewhere in its logging format.
  166. Example use as a context manager:
  167. with borgmatic.logger.Log_prefix('myprefix'):
  168. do_something_that_logs()
  169. For the scope of that "with" statement, any logs created are prefixed with "myprefix: ".
  170. Afterwards, the prefix gets restored to whatever it was prior to the context manager.
  171. '''
  172. def __init__(self, prefix):
  173. '''
  174. Given the desired log prefix, save it for use below. Set prefix to None to disable any
  175. prefix from getting logged.
  176. '''
  177. self.prefix = prefix
  178. self.original_prefix = None
  179. def __enter__(self):
  180. '''
  181. Set the prefix onto the formatter defaults for every logging handler so that the prefix ends
  182. up in every log message. But first, save off any original prefix so that it can be restored
  183. below.
  184. '''
  185. self.original_prefix = get_log_prefix()
  186. set_log_prefix(self.prefix)
  187. def __exit__(self, exception_type, exception, traceback):
  188. '''
  189. Restore any original prefix.
  190. '''
  191. set_log_prefix(self.original_prefix)
  192. class Delayed_logging_handler(logging.handlers.BufferingHandler):
  193. '''
  194. A logging handler that buffers logs and doesn't flush them until explicitly flushed (after
  195. target handlers are actually set). It's useful for holding onto messages logged before logging
  196. is configured, ensuring those records eventually make their way to the relevant logging
  197. handlers.
  198. When flushing, don't forward log records to a target handler if the record's log level is below
  199. that of the handler. This recreates the standard logging behavior of, say, logging.DEBUG records
  200. getting suppressed if a handler's level is only set to logging.INFO.
  201. '''
  202. def __init__(self):
  203. super().__init__(capacity=0)
  204. self.targets = None
  205. def shouldFlush(self, record): # noqa: N802
  206. return self.targets is not None
  207. def flush(self):
  208. self.acquire()
  209. try:
  210. if not self.targets:
  211. return
  212. for record in self.buffer:
  213. for target in self.targets:
  214. if record.levelno >= target.level:
  215. target.handle(record)
  216. self.buffer.clear()
  217. finally:
  218. self.release()
  219. def configure_delayed_logging(): # pragma: no cover
  220. '''
  221. Configure a delayed logging handler to buffer anything that gets logged until we're ready to
  222. deal with it.
  223. '''
  224. logging.basicConfig(
  225. level=logging.DEBUG,
  226. handlers=[Delayed_logging_handler()],
  227. )
  228. def flush_delayed_logging(target_handlers):
  229. '''
  230. Flush any previously buffered logs to our "real" logging handlers.
  231. '''
  232. root_logger = logging.getLogger()
  233. if root_logger.handlers and isinstance(root_logger.handlers[0], Delayed_logging_handler):
  234. delayed_handler = root_logger.handlers[0]
  235. delayed_handler.targets = target_handlers
  236. delayed_handler.flush()
  237. root_logger.removeHandler(delayed_handler)
  238. def configure_logging(
  239. console_log_level,
  240. syslog_log_level=None,
  241. log_file_log_level=None,
  242. monitoring_log_level=None,
  243. log_file=None,
  244. log_file_format=None,
  245. color_enabled=True,
  246. ):
  247. '''
  248. Configure logging to go to both the console and (syslog or log file). Use the given log levels,
  249. respectively. If color is enabled, set up log formatting accordingly.
  250. Raise FileNotFoundError or PermissionError if the log file could not be opened for writing.
  251. '''
  252. add_custom_log_levels()
  253. if syslog_log_level is None:
  254. syslog_log_level = logging.DISABLED
  255. if log_file_log_level is None:
  256. log_file_log_level = console_log_level
  257. if monitoring_log_level is None:
  258. monitoring_log_level = console_log_level
  259. # Log certain log levels to console stderr and others to stdout. This supports use cases like
  260. # grepping (non-error) output.
  261. console_disabled = logging.NullHandler()
  262. console_error_handler = logging.StreamHandler(sys.stderr)
  263. console_standard_handler = logging.StreamHandler(sys.stdout)
  264. console_handler = Multi_stream_handler(
  265. {
  266. logging.DISABLED: console_disabled,
  267. logging.CRITICAL: console_error_handler,
  268. logging.ERROR: console_error_handler,
  269. logging.WARNING: console_error_handler,
  270. logging.ANSWER: console_standard_handler,
  271. logging.INFO: console_standard_handler,
  272. logging.DEBUG: console_standard_handler,
  273. },
  274. )
  275. if color_enabled:
  276. console_handler.setFormatter(Console_color_formatter())
  277. else:
  278. console_handler.setFormatter(Log_prefix_formatter())
  279. console_handler.setLevel(console_log_level)
  280. handlers = [console_handler]
  281. if syslog_log_level != logging.DISABLED:
  282. syslog_path = None
  283. if os.path.exists('/dev/log'):
  284. syslog_path = '/dev/log'
  285. elif os.path.exists('/var/run/syslog'):
  286. syslog_path = '/var/run/syslog'
  287. elif os.path.exists('/var/run/log'):
  288. syslog_path = '/var/run/log'
  289. if syslog_path:
  290. syslog_handler = logging.handlers.SysLogHandler(address=syslog_path)
  291. syslog_handler.setFormatter(
  292. Log_prefix_formatter(
  293. 'borgmatic: {levelname} {prefix}{message}',
  294. ),
  295. )
  296. syslog_handler.setLevel(syslog_log_level)
  297. handlers.append(syslog_handler)
  298. if log_file and log_file_log_level != logging.DISABLED:
  299. file_handler = logging.handlers.WatchedFileHandler(log_file)
  300. file_handler.setFormatter(
  301. Log_prefix_formatter(
  302. log_file_format or '[{asctime}] {levelname}: {prefix}{message}',
  303. ),
  304. )
  305. file_handler.setLevel(log_file_log_level)
  306. handlers.append(file_handler)
  307. flush_delayed_logging(handlers)
  308. logging.basicConfig(
  309. level=min(handler.level for handler in handlers),
  310. handlers=handlers,
  311. )