log.py 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253
  1. # -*- coding: utf-8 -*-
  2. """
  3. celery.utils.log
  4. ~~~~~~~~~~~~~~~~
  5. Logging utilities.
  6. """
  7. from __future__ import absolute_import, print_function
  8. import logging
  9. import os
  10. import sys
  11. import threading
  12. import traceback
  13. from contextlib import contextmanager
  14. from billiard import current_process, util as mputil
  15. from kombu.log import get_logger as _get_logger, LOG_LEVELS
  16. from celery.five import string_t
  17. from .encoding import safe_str, str_t
  18. from .term import colored
  19. _process_aware = False
  20. PY3 = sys.version_info[0] == 3
  21. MP_LOG = os.environ.get('MP_LOG', False)
  22. # Sets up our logging hierarchy.
  23. #
  24. # Every logger in the celery package inherits from the "celery"
  25. # logger, and every task logger inherits from the "celery.task"
  26. # logger.
  27. base_logger = logger = _get_logger('celery')
  28. mp_logger = _get_logger('multiprocessing')
  29. _in_sighandler = False
  30. def set_in_sighandler(value):
  31. global _in_sighandler
  32. _in_sighandler = value
  33. @contextmanager
  34. def in_sighandler():
  35. set_in_sighandler(True)
  36. yield
  37. set_in_sighandler(False)
  38. def get_logger(name):
  39. l = _get_logger(name)
  40. if logging.root not in (l, l.parent) and l is not base_logger:
  41. l.parent = base_logger
  42. return l
  43. task_logger = get_logger('celery.task')
  44. worker_logger = get_logger('celery.worker')
  45. def get_task_logger(name):
  46. logger = get_logger(name)
  47. if logger.parent is logging.root:
  48. logger.parent = task_logger
  49. return logger
  50. def mlevel(level):
  51. if level and not isinstance(level, int):
  52. return LOG_LEVELS[level.upper()]
  53. return level
  54. class ColorFormatter(logging.Formatter):
  55. #: Loglevel -> Color mapping.
  56. COLORS = colored().names
  57. colors = {'DEBUG': COLORS['blue'], 'WARNING': COLORS['yellow'],
  58. 'ERROR': COLORS['red'], 'CRITICAL': COLORS['magenta']}
  59. def __init__(self, fmt=None, use_color=True):
  60. logging.Formatter.__init__(self, fmt)
  61. self.use_color = use_color
  62. def formatException(self, ei):
  63. if ei and not isinstance(ei, tuple):
  64. ei = sys.exc_info()
  65. r = logging.Formatter.formatException(self, ei)
  66. if isinstance(r, str) and not PY3:
  67. return safe_str(r)
  68. return r
  69. def format(self, record):
  70. levelname = record.levelname
  71. color = self.colors.get(levelname)
  72. if self.use_color and color:
  73. msg = record.msg
  74. try:
  75. # safe_str will repr the color object
  76. # and color will break on non-string objects
  77. # so need to reorder calls based on type.
  78. # Issue #427
  79. if isinstance(msg, string_t):
  80. record.msg = str_t(color(safe_str(msg)))
  81. else:
  82. record.msg = safe_str(color(msg))
  83. except Exception as exc:
  84. record.msg = '<Unrepresentable {0!r}: {1!r}>'.format(
  85. type(msg), exc)
  86. record.exc_info = True
  87. return safe_str(logging.Formatter.format(self, record))
  88. class LoggingProxy(object):
  89. """Forward file object to :class:`logging.Logger` instance.
  90. :param logger: The :class:`logging.Logger` instance to forward to.
  91. :param loglevel: Loglevel to use when writing messages.
  92. """
  93. mode = 'w'
  94. name = None
  95. closed = False
  96. loglevel = logging.ERROR
  97. _thread = threading.local()
  98. def __init__(self, logger, loglevel=None):
  99. self.logger = logger
  100. self.loglevel = mlevel(loglevel or self.logger.level or self.loglevel)
  101. self._safewrap_handlers()
  102. def _safewrap_handlers(self):
  103. """Make the logger handlers dump internal errors to
  104. `sys.__stderr__` instead of `sys.stderr` to circumvent
  105. infinite loops."""
  106. def wrap_handler(handler): # pragma: no cover
  107. class WithSafeHandleError(logging.Handler):
  108. def handleError(self, record):
  109. exc_info = sys.exc_info()
  110. try:
  111. try:
  112. traceback.print_exception(exc_info[0],
  113. exc_info[1],
  114. exc_info[2],
  115. None, sys.__stderr__)
  116. except IOError:
  117. pass # see python issue 5971
  118. finally:
  119. del(exc_info)
  120. handler.handleError = WithSafeHandleError().handleError
  121. return [wrap_handler(h) for h in self.logger.handlers]
  122. def write(self, data):
  123. """Write message to logging object."""
  124. if _in_sighandler:
  125. print(safe_str(data), file=sys.__stderr__)
  126. if getattr(self._thread, 'recurse_protection', False):
  127. # Logger is logging back to this file, so stop recursing.
  128. return
  129. data = data.strip()
  130. if data and not self.closed:
  131. self._thread.recurse_protection = True
  132. try:
  133. self.logger.log(self.loglevel, safe_str(data))
  134. finally:
  135. self._thread.recurse_protection = False
  136. def writelines(self, sequence):
  137. """`writelines(sequence_of_strings) -> None`.
  138. Write the strings to the file.
  139. The sequence can be any iterable object producing strings.
  140. This is equivalent to calling :meth:`write` for each string.
  141. """
  142. for part in sequence:
  143. self.write(part)
  144. def flush(self):
  145. """This object is not buffered so any :meth:`flush` requests
  146. are ignored."""
  147. pass
  148. def close(self):
  149. """When the object is closed, no write requests are forwarded to
  150. the logging object anymore."""
  151. self.closed = True
  152. def isatty(self):
  153. """Always returns :const:`False`. Just here for file support."""
  154. return False
  155. def ensure_process_aware_logger():
  156. """Make sure process name is recorded when loggers are used."""
  157. global _process_aware
  158. if not _process_aware:
  159. logging._acquireLock()
  160. try:
  161. _process_aware = True
  162. Logger = logging.getLoggerClass()
  163. if getattr(Logger, '_process_aware', False): # pragma: no cover
  164. return
  165. class ProcessAwareLogger(Logger):
  166. _process_aware = True
  167. def makeRecord(self, *args, **kwds):
  168. record = Logger.makeRecord(self, *args, **kwds)
  169. record.processName = current_process()._name
  170. return record
  171. logging.setLoggerClass(ProcessAwareLogger)
  172. finally:
  173. logging._releaseLock()
  174. def get_multiprocessing_logger():
  175. return mputil.get_logger() if mputil else None
  176. def reset_multiprocessing_logger():
  177. if mputil and hasattr(mputil, '_logger'):
  178. mputil._logger = None
  179. def _patch_logger_class():
  180. """Make sure loggers don't log while in a signal handler."""
  181. logging._acquireLock()
  182. try:
  183. OldLoggerClass = logging.getLoggerClass()
  184. if not getattr(OldLoggerClass, '_signal_safe', False):
  185. class SigSafeLogger(OldLoggerClass):
  186. _signal_safe = True
  187. def log(self, *args, **kwargs):
  188. if _in_sighandler:
  189. return
  190. return OldLoggerClass.log(self, *args, **kwargs)
  191. logging.setLoggerClass(SigSafeLogger)
  192. finally:
  193. logging._releaseLock()
  194. _patch_logger_class()