utils.py 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. # -*- coding: utf-8 -*-
  2. """
  3. celery.app.utils
  4. ~~~~~~~~~~~~~~~~
  5. App utilities: Compat settings, bugreport tool, pickling apps.
  6. """
  7. from __future__ import absolute_import
  8. import os
  9. import platform as _platform
  10. import re
  11. from collections import Mapping
  12. from celery.datastructures import ConfigurationView
  13. from celery.five import items, string_t
  14. from celery.platforms import pyimplementation
  15. from celery.utils.text import pretty
  16. from celery.utils.imports import qualname
  17. from .defaults import find
  18. __all__ = ['appstr', 'Settings', 'filter_hidden_settings', 'bugreport']
  19. #: Format used to generate bugreport information.
  20. BUGREPORT_INFO = """
  21. software -> celery:{celery_v} kombu:{kombu_v} py:{py_v}
  22. billiard:{billiard_v} {driver_v}
  23. platform -> system:{system} arch:{arch} imp:{py_i}
  24. loader -> {loader}
  25. settings -> transport:{transport} results:{results}
  26. {human_settings}
  27. """
  28. HIDDEN_SETTINGS = re.compile(
  29. 'API|TOKEN|KEY|SECRET|PASS|PROFANITIES_LIST|SIGNATURE|DATABASE',
  30. re.IGNORECASE,
  31. )
  32. def appstr(app):
  33. """String used in __repr__ etc, to id app instances."""
  34. return '{0}:0x{1:x}'.format(app.main or '__main__', id(app))
  35. class Settings(ConfigurationView):
  36. """Celery settings object."""
  37. @property
  38. def CELERY_RESULT_BACKEND(self):
  39. return self.first('CELERY_RESULT_BACKEND', 'CELERY_BACKEND')
  40. @property
  41. def BROKER_TRANSPORT(self):
  42. return self.first('BROKER_TRANSPORT',
  43. 'BROKER_BACKEND', 'CARROT_BACKEND')
  44. @property
  45. def BROKER_BACKEND(self):
  46. """Deprecated compat alias to :attr:`BROKER_TRANSPORT`."""
  47. return self.BROKER_TRANSPORT
  48. @property
  49. def BROKER_URL(self):
  50. return (os.environ.get('CELERY_BROKER_URL') or
  51. self.first('BROKER_URL', 'BROKER_HOST'))
  52. @property
  53. def CELERY_TIMEZONE(self):
  54. # this way we also support django's time zone.
  55. return self.first('CELERY_TIMEZONE', 'TIME_ZONE')
  56. def without_defaults(self):
  57. """Return the current configuration, but without defaults."""
  58. # the last stash is the default settings, so just skip that
  59. return Settings({}, self._order[:-1])
  60. def value_set_for(self, key):
  61. return key in self.without_defaults()
  62. def find_option(self, name, namespace='celery'):
  63. """Search for option by name.
  64. Will return ``(namespace, key, type)`` tuple, e.g.::
  65. >>> app.conf.find_option('disable_rate_limits')
  66. ('CELERY', 'DISABLE_RATE_LIMITS',
  67. <Option: type->bool default->False>))
  68. :param name: Name of option, cannot be partial.
  69. :keyword namespace: Preferred namespace (``CELERY`` by default).
  70. """
  71. return find(name, namespace)
  72. def find_value_for_key(self, name, namespace='celery'):
  73. """Shortcut to ``get_by_parts(*find_option(name)[:-1])``"""
  74. return self.get_by_parts(*self.find_option(name, namespace)[:-1])
  75. def get_by_parts(self, *parts):
  76. """Return the current value for setting specified as a path.
  77. Example::
  78. >>> celery.conf.get_by_parts('CELERY', 'DISABLE_RATE_LIMITS')
  79. False
  80. """
  81. return self['_'.join(part for part in parts if part)]
  82. def table(self, with_defaults=False, censored=True):
  83. filt = filter_hidden_settings if censored else lambda v: v
  84. return filt(dict(
  85. (k, v) for k, v in items(
  86. self if with_defaults else self.without_defaults())
  87. if k.isupper() and not k.startswith('_')
  88. ))
  89. def humanize(self, with_defaults=False, censored=True):
  90. """Return a human readable string showing changes to the
  91. configuration."""
  92. return '\n'.join(
  93. '{0}: {1}'.format(key, pretty(value, width=50))
  94. for key, value in items(self.table(with_defaults, censored)))
  95. class AppPickler(object):
  96. """Old application pickler/unpickler (< 3.1)."""
  97. def __call__(self, cls, *args):
  98. kwargs = self.build_kwargs(*args)
  99. app = self.construct(cls, **kwargs)
  100. self.prepare(app, **kwargs)
  101. return app
  102. def prepare(self, app, **kwargs):
  103. app.conf.update(kwargs['changes'])
  104. def build_kwargs(self, *args):
  105. return self.build_standard_kwargs(*args)
  106. def build_standard_kwargs(self, main, changes, loader, backend, amqp,
  107. events, log, control, accept_magic_kwargs,
  108. config_source=None):
  109. return dict(main=main, loader=loader, backend=backend, amqp=amqp,
  110. changes=changes, events=events, log=log, control=control,
  111. set_as_current=False,
  112. accept_magic_kwargs=accept_magic_kwargs,
  113. config_source=config_source)
  114. def construct(self, cls, **kwargs):
  115. return cls(**kwargs)
  116. def _unpickle_app(cls, pickler, *args):
  117. """Rebuild app for versions 2.5+"""
  118. return pickler()(cls, *args)
  119. def _unpickle_app_v2(cls, kwargs):
  120. """Rebuild app for versions 3.1+"""
  121. kwargs['set_as_current'] = False
  122. return cls(**kwargs)
  123. def filter_hidden_settings(conf):
  124. def maybe_censor(key, value, mask='*' * 8):
  125. if isinstance(value, Mapping):
  126. return filter_hidden_settings(value)
  127. if isinstance(value, string_t) and HIDDEN_SETTINGS.search(key):
  128. return mask
  129. if 'BROKER_URL' in key.upper():
  130. from kombu import Connection
  131. return Connection(value).as_uri(mask=mask)
  132. return value
  133. return dict((k, maybe_censor(k, v)) for k, v in items(conf))
  134. def bugreport(app):
  135. """Return a string containing information useful in bug reports."""
  136. import billiard
  137. import celery
  138. import kombu
  139. try:
  140. conn = app.connection()
  141. driver_v = '{0}:{1}'.format(conn.transport.driver_name,
  142. conn.transport.driver_version())
  143. transport = conn.transport_cls
  144. except Exception:
  145. transport = driver_v = ''
  146. return BUGREPORT_INFO.format(
  147. system=_platform.system(),
  148. arch=', '.join(x for x in _platform.architecture() if x),
  149. py_i=pyimplementation(),
  150. celery_v=celery.VERSION_BANNER,
  151. kombu_v=kombu.__version__,
  152. billiard_v=billiard.__version__,
  153. py_v=_platform.python_version(),
  154. driver_v=driver_v,
  155. transport=transport,
  156. results=app.conf.CELERY_RESULT_BACKEND or 'disabled',
  157. human_settings=app.conf.humanize(),
  158. loader=qualname(app.loader.__class__),
  159. )