compat.py 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. # -*- coding: utf-8 -*-
  2. """
  3. celery.utils.compat
  4. ~~~~~~~~~~~~~~~~~~~
  5. Backward compatible implementations of features
  6. only available in newer Python versions.
  7. :copyright: (c) 2009 - 2012 by Ask Solem.
  8. :license: BSD, see LICENSE for more details.
  9. """
  10. from __future__ import absolute_import
  11. ############## py3k #########################################################
  12. import sys
  13. try:
  14. reload = reload # noqa
  15. except NameError:
  16. from imp import reload # noqa
  17. try:
  18. from UserList import UserList # noqa
  19. except ImportError:
  20. from collections import UserList # noqa
  21. try:
  22. from UserDict import UserDict # noqa
  23. except ImportError:
  24. from collections import UserDict # noqa
  25. if sys.version_info >= (3, 0):
  26. from io import StringIO, BytesIO
  27. from .encoding import bytes_to_str
  28. class WhateverIO(StringIO):
  29. def write(self, data):
  30. StringIO.write(self, bytes_to_str(data))
  31. else:
  32. try:
  33. from cStringIO import StringIO # noqa
  34. except ImportError:
  35. from StringIO import StringIO # noqa
  36. BytesIO = WhateverIO = StringIO # noqa
  37. ############## collections.OrderedDict ######################################
  38. try:
  39. from collections import OrderedDict
  40. except ImportError:
  41. from ordereddict import OrderedDict # noqa
  42. ############## logging.LoggerAdapter ########################################
  43. import logging
  44. try:
  45. import multiprocessing
  46. except ImportError:
  47. multiprocessing = None # noqa
  48. import sys
  49. def _checkLevel(level):
  50. if isinstance(level, int):
  51. rv = level
  52. elif str(level) == level:
  53. if level not in logging._levelNames:
  54. raise ValueError("Unknown level: %r" % level)
  55. rv = logging._levelNames[level]
  56. else:
  57. raise TypeError("Level not an integer or a valid string: %r" % level)
  58. return rv
  59. class _CompatLoggerAdapter(object):
  60. def __init__(self, logger, extra):
  61. self.logger = logger
  62. self.extra = extra
  63. def setLevel(self, level):
  64. self.logger.level = _checkLevel(level)
  65. def process(self, msg, kwargs):
  66. kwargs["extra"] = self.extra
  67. return msg, kwargs
  68. def debug(self, msg, *args, **kwargs):
  69. self.log(logging.DEBUG, msg, *args, **kwargs)
  70. def info(self, msg, *args, **kwargs):
  71. self.log(logging.INFO, msg, *args, **kwargs)
  72. def warning(self, msg, *args, **kwargs):
  73. self.log(logging.WARNING, msg, *args, **kwargs)
  74. warn = warning
  75. def error(self, msg, *args, **kwargs):
  76. self.log(logging.ERROR, msg, *args, **kwargs)
  77. def exception(self, msg, *args, **kwargs):
  78. kwargs.setdefault("exc_info", 1)
  79. self.error(msg, *args, **kwargs)
  80. def critical(self, msg, *args, **kwargs):
  81. self.log(logging.CRITICAL, msg, *args, **kwargs)
  82. fatal = critical
  83. def log(self, level, msg, *args, **kwargs):
  84. if self.logger.isEnabledFor(level):
  85. msg, kwargs = self.process(msg, kwargs)
  86. self._log(level, msg, args, **kwargs)
  87. def makeRecord(self, name, level, fn, lno, msg, args, exc_info,
  88. func=None, extra=None):
  89. rv = logging.LogRecord(name, level, fn, lno, msg, args, exc_info, func)
  90. if extra is not None:
  91. for key, value in extra.items():
  92. if key in ("message", "asctime") or key in rv.__dict__:
  93. raise KeyError(
  94. "Attempt to override %r in LogRecord" % key)
  95. rv.__dict__[key] = value
  96. if multiprocessing is not None:
  97. rv.processName = multiprocessing.current_process()._name
  98. else:
  99. rv.processName = ""
  100. return rv
  101. def _log(self, level, msg, args, exc_info=None, extra=None):
  102. defcaller = "(unknown file)", 0, "(unknown function)"
  103. if logging._srcfile:
  104. # IronPython doesn't track Python frames, so findCaller
  105. # throws an exception on some versions of IronPython.
  106. # We trap it here so that IronPython can use logging.
  107. try:
  108. fn, lno, func = self.logger.findCaller()
  109. except ValueError:
  110. fn, lno, func = defcaller
  111. else:
  112. fn, lno, func = defcaller
  113. if exc_info:
  114. if not isinstance(exc_info, tuple):
  115. exc_info = sys.exc_info()
  116. record = self.makeRecord(self.logger.name, level, fn, lno, msg,
  117. args, exc_info, func, extra)
  118. self.logger.handle(record)
  119. def isEnabledFor(self, level):
  120. return self.logger.isEnabledFor(level)
  121. def addHandler(self, hdlr):
  122. self.logger.addHandler(hdlr)
  123. def removeHandler(self, hdlr):
  124. self.logger.removeHandler(hdlr)
  125. @property
  126. def level(self):
  127. return self.logger.level
  128. try:
  129. from logging import LoggerAdapter
  130. except ImportError:
  131. LoggerAdapter = _CompatLoggerAdapter # noqa
  132. ############## itertools.zip_longest #######################################
  133. try:
  134. from itertools import izip_longest as zip_longest
  135. except ImportError:
  136. import itertools
  137. def zip_longest(*args, **kwds): # noqa
  138. fillvalue = kwds.get("fillvalue")
  139. def sentinel(counter=([fillvalue] * (len(args) - 1)).pop):
  140. yield counter() # yields the fillvalue, or raises IndexError
  141. fillers = itertools.repeat(fillvalue)
  142. iters = [itertools.chain(it, sentinel(), fillers)
  143. for it in args]
  144. try:
  145. for tup in itertools.izip(*iters):
  146. yield tup
  147. except IndexError:
  148. pass
  149. ############## itertools.chain.from_iterable ################################
  150. from itertools import chain
  151. def _compat_chain_from_iterable(iterables):
  152. for it in iterables:
  153. for element in it:
  154. yield element
  155. try:
  156. chain_from_iterable = getattr(chain, "from_iterable")
  157. except AttributeError:
  158. chain_from_iterable = _compat_chain_from_iterable
  159. ############## logging.handlers.WatchedFileHandler ##########################
  160. import os
  161. from stat import ST_DEV, ST_INO
  162. import platform as _platform
  163. if _platform.system() == "Windows":
  164. #since windows doesn't go with WatchedFileHandler use FileHandler instead
  165. WatchedFileHandler = logging.FileHandler
  166. else:
  167. try:
  168. from logging.handlers import WatchedFileHandler
  169. except ImportError:
  170. class WatchedFileHandler(logging.FileHandler): # noqa
  171. """
  172. A handler for logging to a file, which watches the file
  173. to see if it has changed while in use. This can happen because of
  174. usage of programs such as newsyslog and logrotate which perform
  175. log file rotation. This handler, intended for use under Unix,
  176. watches the file to see if it has changed since the last emit.
  177. (A file has changed if its device or inode have changed.)
  178. If it has changed, the old file stream is closed, and the file
  179. opened to get a new stream.
  180. This handler is not appropriate for use under Windows, because
  181. under Windows open files cannot be moved or renamed - logging
  182. opens the files with exclusive locks - and so there is no need
  183. for such a handler. Furthermore, ST_INO is not supported under
  184. Windows; stat always returns zero for this value.
  185. This handler is based on a suggestion and patch by Chad J.
  186. Schroeder.
  187. """
  188. def __init__(self, *args, **kwargs):
  189. logging.FileHandler.__init__(self, *args, **kwargs)
  190. if not os.path.exists(self.baseFilename):
  191. self.dev, self.ino = -1, -1
  192. else:
  193. stat = os.stat(self.baseFilename)
  194. self.dev, self.ino = stat[ST_DEV], stat[ST_INO]
  195. def emit(self, record):
  196. """
  197. Emit a record.
  198. First check if the underlying file has changed, and if it
  199. has, close the old stream and reopen the file to get the
  200. current stream.
  201. """
  202. if not os.path.exists(self.baseFilename):
  203. stat = None
  204. changed = 1
  205. else:
  206. stat = os.stat(self.baseFilename)
  207. changed = ((stat[ST_DEV] != self.dev) or
  208. (stat[ST_INO] != self.ino))
  209. if changed and self.stream is not None:
  210. self.stream.flush()
  211. self.stream.close()
  212. self.stream = self._open()
  213. if stat is None:
  214. stat = os.stat(self.baseFilename)
  215. self.dev, self.ino = stat[ST_DEV], stat[ST_INO]
  216. logging.FileHandler.emit(self, record)