log.py 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244
  1. # -*- coding: utf-8 -*-
  2. """
  3. celery.app.log
  4. ~~~~~~~~~~~~~~
  5. The Celery instances logging section: ``Celery.log``.
  6. Sets up logging for the worker and other programs,
  7. redirects stdouts, colors log output, patches logging
  8. related compatibility fixes, and so on.
  9. """
  10. from __future__ import absolute_import
  11. import logging
  12. import os
  13. import sys
  14. from logging.handlers import WatchedFileHandler
  15. from kombu.log import NullHandler
  16. from celery import signals
  17. from celery._state import get_current_task
  18. from celery.five import class_property, string_t
  19. from celery.utils import isatty
  20. from celery.utils.log import (
  21. get_logger, mlevel,
  22. ColorFormatter, ensure_process_aware_logger,
  23. LoggingProxy, get_multiprocessing_logger,
  24. reset_multiprocessing_logger,
  25. )
  26. from celery.utils.term import colored
  27. __all__ = ['TaskFormatter', 'Logging']
  28. MP_LOG = os.environ.get('MP_LOG', False)
  29. class TaskFormatter(ColorFormatter):
  30. def format(self, record):
  31. task = get_current_task()
  32. if task and task.request:
  33. record.__dict__.update(task_id=task.request.id,
  34. task_name=task.name)
  35. else:
  36. record.__dict__.setdefault('task_name', '???')
  37. record.__dict__.setdefault('task_id', '???')
  38. return ColorFormatter.format(self, record)
  39. class Logging(object):
  40. #: The logging subsystem is only configured once per process.
  41. #: setup_logging_subsystem sets this flag, and subsequent calls
  42. #: will do nothing.
  43. _setup = False
  44. def __init__(self, app):
  45. self.app = app
  46. self.loglevel = mlevel(self.app.conf.CELERYD_LOG_LEVEL)
  47. self.format = self.app.conf.CELERYD_LOG_FORMAT
  48. self.task_format = self.app.conf.CELERYD_TASK_LOG_FORMAT
  49. self.colorize = self.app.conf.CELERYD_LOG_COLOR
  50. def setup(self, loglevel=None, logfile=None, redirect_stdouts=False,
  51. redirect_level='WARNING', colorize=None):
  52. handled = self.setup_logging_subsystem(
  53. loglevel, logfile, colorize=colorize,
  54. )
  55. if not handled:
  56. if redirect_stdouts:
  57. self.redirect_stdouts(redirect_level)
  58. os.environ.update(
  59. CELERY_LOG_LEVEL=str(loglevel) if loglevel else '',
  60. CELERY_LOG_FILE=str(logfile) if logfile else '',
  61. )
  62. return handled
  63. def redirect_stdouts(self, loglevel=None, name='celery.redirected'):
  64. self.redirect_stdouts_to_logger(
  65. get_logger(name), loglevel=loglevel
  66. )
  67. os.environ.update(
  68. CELERY_LOG_REDIRECT='1',
  69. CELERY_LOG_REDIRECT_LEVEL=str(loglevel or ''),
  70. )
  71. def setup_logging_subsystem(self, loglevel=None, logfile=None,
  72. format=None, colorize=None, **kwargs):
  73. if self.already_setup:
  74. return
  75. self.already_setup = True
  76. loglevel = mlevel(loglevel or self.loglevel)
  77. format = format or self.format
  78. colorize = self.supports_color(colorize, logfile)
  79. reset_multiprocessing_logger()
  80. ensure_process_aware_logger()
  81. receivers = signals.setup_logging.send(
  82. sender=None, loglevel=loglevel, logfile=logfile,
  83. format=format, colorize=colorize,
  84. )
  85. if not receivers:
  86. root = logging.getLogger()
  87. if self.app.conf.CELERYD_HIJACK_ROOT_LOGGER:
  88. root.handlers = []
  89. # Configure root logger
  90. self._configure_logger(
  91. root, logfile, loglevel, format, colorize, **kwargs
  92. )
  93. # Configure the multiprocessing logger
  94. self._configure_logger(
  95. get_multiprocessing_logger(),
  96. logfile, loglevel if MP_LOG else logging.ERROR,
  97. format, colorize, **kwargs
  98. )
  99. signals.after_setup_logger.send(
  100. sender=None, logger=root,
  101. loglevel=loglevel, logfile=logfile,
  102. format=format, colorize=colorize,
  103. )
  104. # then setup the root task logger.
  105. self.setup_task_loggers(loglevel, logfile, colorize=colorize)
  106. # This is a hack for multiprocessing's fork+exec, so that
  107. # logging before Process.run works.
  108. logfile_name = logfile if isinstance(logfile, string_t) else ''
  109. os.environ.update(_MP_FORK_LOGLEVEL_=str(loglevel),
  110. _MP_FORK_LOGFILE_=logfile_name,
  111. _MP_FORK_LOGFORMAT_=format)
  112. return receivers
  113. def _configure_logger(self, logger, logfile, loglevel,
  114. format, colorize, **kwargs):
  115. if logger is not None:
  116. self.setup_handlers(logger, logfile, format,
  117. colorize, **kwargs)
  118. if loglevel:
  119. logger.setLevel(loglevel)
  120. def setup_task_loggers(self, loglevel=None, logfile=None, format=None,
  121. colorize=None, propagate=False, **kwargs):
  122. """Setup the task logger.
  123. If `logfile` is not specified, then `sys.stderr` is used.
  124. Will return the base task logger object.
  125. """
  126. loglevel = mlevel(loglevel or self.loglevel)
  127. format = format or self.task_format
  128. colorize = self.supports_color(colorize, logfile)
  129. logger = self.setup_handlers(
  130. get_logger('celery.task'),
  131. logfile, format, colorize,
  132. formatter=TaskFormatter, **kwargs
  133. )
  134. logger.setLevel(loglevel)
  135. logger.propagate = int(propagate) # this is an int for some reason.
  136. # better to not question why.
  137. signals.after_setup_task_logger.send(
  138. sender=None, logger=logger,
  139. loglevel=loglevel, logfile=logfile,
  140. format=format, colorize=colorize,
  141. )
  142. return logger
  143. def redirect_stdouts_to_logger(self, logger, loglevel=None,
  144. stdout=True, stderr=True):
  145. """Redirect :class:`sys.stdout` and :class:`sys.stderr` to a
  146. logging instance.
  147. :param logger: The :class:`logging.Logger` instance to redirect to.
  148. :param loglevel: The loglevel redirected messages will be logged as.
  149. """
  150. proxy = LoggingProxy(logger, loglevel)
  151. if stdout:
  152. sys.stdout = proxy
  153. if stderr:
  154. sys.stderr = proxy
  155. return proxy
  156. def supports_color(self, colorize=None, logfile=None):
  157. colorize = self.colorize if colorize is None else colorize
  158. if self.app.IS_WINDOWS:
  159. # Windows does not support ANSI color codes.
  160. return False
  161. if colorize or colorize is None:
  162. # Only use color if there is no active log file
  163. # and stderr is an actual terminal.
  164. return logfile is None and isatty(sys.stderr)
  165. return colorize
  166. def colored(self, logfile=None, enabled=None):
  167. return colored(enabled=self.supports_color(enabled, logfile))
  168. def setup_handlers(self, logger, logfile, format, colorize,
  169. formatter=ColorFormatter, **kwargs):
  170. if self._is_configured(logger):
  171. return logger
  172. handler = self._detect_handler(logfile)
  173. handler.setFormatter(formatter(format, use_color=colorize))
  174. logger.addHandler(handler)
  175. return logger
  176. def _detect_handler(self, logfile=None):
  177. """Create log handler with either a filename, an open stream
  178. or :const:`None` (stderr)."""
  179. logfile = sys.__stderr__ if logfile is None else logfile
  180. if hasattr(logfile, 'write'):
  181. return logging.StreamHandler(logfile)
  182. return WatchedFileHandler(logfile)
  183. def _has_handler(self, logger):
  184. return (logger.handlers and
  185. not isinstance(logger.handlers[0], NullHandler))
  186. def _is_configured(self, logger):
  187. return self._has_handler(logger) and not getattr(
  188. logger, '_rudimentary_setup', False)
  189. def setup_logger(self, name='celery', *args, **kwargs):
  190. """Deprecated: No longer used."""
  191. self.setup_logging_subsystem(*args, **kwargs)
  192. return logging.root
  193. def get_default_logger(self, name='celery', **kwargs):
  194. return get_logger(name)
  195. @class_property
  196. def already_setup(cls):
  197. return cls._setup
  198. @already_setup.setter # noqa
  199. def already_setup(cls, was_setup):
  200. cls._setup = was_setup