compat.py 8.1 KB

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