Development of an internal social media platform with personalised dashboards for students
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

base.py 9.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299
  1. # -*- coding: utf-8 -*-
  2. """
  3. celery.loaders.base
  4. ~~~~~~~~~~~~~~~~~~~
  5. Loader base class.
  6. """
  7. from __future__ import absolute_import
  8. import anyjson
  9. import imp as _imp
  10. import importlib
  11. import os
  12. import re
  13. import sys
  14. from datetime import datetime
  15. from kombu.utils import cached_property
  16. from kombu.utils.encoding import safe_str
  17. from celery import signals
  18. from celery.datastructures import DictAttribute, force_mapping
  19. from celery.five import reraise, string_t
  20. from celery.utils.functional import maybe_list
  21. from celery.utils.imports import (
  22. import_from_cwd, symbol_by_name, NotAPackage, find_module,
  23. )
  24. __all__ = ['BaseLoader']
  25. _RACE_PROTECTION = False
  26. CONFIG_INVALID_NAME = """\
  27. Error: Module '{module}' doesn't exist, or it's not a valid \
  28. Python module name.
  29. """
  30. CONFIG_WITH_SUFFIX = CONFIG_INVALID_NAME + """\
  31. Did you mean '{suggest}'?
  32. """
  33. class BaseLoader(object):
  34. """The base class for loaders.
  35. Loaders handles,
  36. * Reading celery client/worker configurations.
  37. * What happens when a task starts?
  38. See :meth:`on_task_init`.
  39. * What happens when the worker starts?
  40. See :meth:`on_worker_init`.
  41. * What happens when the worker shuts down?
  42. See :meth:`on_worker_shutdown`.
  43. * What modules are imported to find tasks?
  44. """
  45. builtin_modules = frozenset()
  46. configured = False
  47. override_backends = {}
  48. worker_initialized = False
  49. _conf = None
  50. def __init__(self, app, **kwargs):
  51. self.app = app
  52. self.task_modules = set()
  53. def now(self, utc=True):
  54. if utc:
  55. return datetime.utcnow()
  56. return datetime.now()
  57. def on_task_init(self, task_id, task):
  58. """This method is called before a task is executed."""
  59. pass
  60. def on_process_cleanup(self):
  61. """This method is called after a task is executed."""
  62. pass
  63. def on_worker_init(self):
  64. """This method is called when the worker (:program:`celery worker`)
  65. starts."""
  66. pass
  67. def on_worker_shutdown(self):
  68. """This method is called when the worker (:program:`celery worker`)
  69. shuts down."""
  70. pass
  71. def on_worker_process_init(self):
  72. """This method is called when a child process starts."""
  73. pass
  74. def import_task_module(self, module):
  75. self.task_modules.add(module)
  76. return self.import_from_cwd(module)
  77. def import_module(self, module, package=None):
  78. return importlib.import_module(module, package=package)
  79. def import_from_cwd(self, module, imp=None, package=None):
  80. return import_from_cwd(
  81. module,
  82. self.import_module if imp is None else imp,
  83. package=package,
  84. )
  85. def import_default_modules(self):
  86. signals.import_modules.send(sender=self.app)
  87. return [
  88. self.import_task_module(m) for m in (
  89. tuple(self.builtin_modules) +
  90. tuple(maybe_list(self.app.conf.CELERY_IMPORTS)) +
  91. tuple(maybe_list(self.app.conf.CELERY_INCLUDE))
  92. )
  93. ]
  94. def init_worker(self):
  95. if not self.worker_initialized:
  96. self.worker_initialized = True
  97. self.import_default_modules()
  98. self.on_worker_init()
  99. def shutdown_worker(self):
  100. self.on_worker_shutdown()
  101. def init_worker_process(self):
  102. self.on_worker_process_init()
  103. def config_from_object(self, obj, silent=False):
  104. if isinstance(obj, string_t):
  105. try:
  106. obj = self._smart_import(obj, imp=self.import_from_cwd)
  107. except (ImportError, AttributeError):
  108. if silent:
  109. return False
  110. raise
  111. self._conf = force_mapping(obj)
  112. return True
  113. def _smart_import(self, path, imp=None):
  114. imp = self.import_module if imp is None else imp
  115. if ':' in path:
  116. # Path includes attribute so can just jump here.
  117. # e.g. ``os.path:abspath``.
  118. return symbol_by_name(path, imp=imp)
  119. # Not sure if path is just a module name or if it includes an
  120. # attribute name (e.g. ``os.path``, vs, ``os.path.abspath``).
  121. try:
  122. return imp(path)
  123. except ImportError:
  124. # Not a module name, so try module + attribute.
  125. return symbol_by_name(path, imp=imp)
  126. def _import_config_module(self, name):
  127. try:
  128. self.find_module(name)
  129. except NotAPackage:
  130. if name.endswith('.py'):
  131. reraise(NotAPackage, NotAPackage(CONFIG_WITH_SUFFIX.format(
  132. module=name, suggest=name[:-3])), sys.exc_info()[2])
  133. reraise(NotAPackage, NotAPackage(CONFIG_INVALID_NAME.format(
  134. module=name)), sys.exc_info()[2])
  135. else:
  136. return self.import_from_cwd(name)
  137. def find_module(self, module):
  138. return find_module(module)
  139. def cmdline_config_parser(
  140. self, args, namespace='celery',
  141. re_type=re.compile(r'\((\w+)\)'),
  142. extra_types={'json': anyjson.loads},
  143. override_types={'tuple': 'json',
  144. 'list': 'json',
  145. 'dict': 'json'}):
  146. from celery.app.defaults import Option, NAMESPACES
  147. namespace = namespace.upper()
  148. typemap = dict(Option.typemap, **extra_types)
  149. def getarg(arg):
  150. """Parse a single configuration definition from
  151. the command-line."""
  152. # ## find key/value
  153. # ns.key=value|ns_key=value (case insensitive)
  154. key, value = arg.split('=', 1)
  155. key = key.upper().replace('.', '_')
  156. # ## find namespace.
  157. # .key=value|_key=value expands to default namespace.
  158. if key[0] == '_':
  159. ns, key = namespace, key[1:]
  160. else:
  161. # find namespace part of key
  162. ns, key = key.split('_', 1)
  163. ns_key = (ns and ns + '_' or '') + key
  164. # (type)value makes cast to custom type.
  165. cast = re_type.match(value)
  166. if cast:
  167. type_ = cast.groups()[0]
  168. type_ = override_types.get(type_, type_)
  169. value = value[len(cast.group()):]
  170. value = typemap[type_](value)
  171. else:
  172. try:
  173. value = NAMESPACES[ns][key].to_python(value)
  174. except ValueError as exc:
  175. # display key name in error message.
  176. raise ValueError('{0!r}: {1}'.format(ns_key, exc))
  177. return ns_key, value
  178. return dict(getarg(arg) for arg in args)
  179. def mail_admins(self, subject, body, fail_silently=False,
  180. sender=None, to=None, host=None, port=None,
  181. user=None, password=None, timeout=None,
  182. use_ssl=False, use_tls=False, charset='utf-8'):
  183. message = self.mail.Message(sender=sender, to=to,
  184. subject=safe_str(subject),
  185. body=safe_str(body),
  186. charset=charset)
  187. mailer = self.mail.Mailer(host=host, port=port,
  188. user=user, password=password,
  189. timeout=timeout, use_ssl=use_ssl,
  190. use_tls=use_tls)
  191. mailer.send(message, fail_silently=fail_silently)
  192. def read_configuration(self, env='CELERY_CONFIG_MODULE'):
  193. try:
  194. custom_config = os.environ[env]
  195. except KeyError:
  196. pass
  197. else:
  198. if custom_config:
  199. usercfg = self._import_config_module(custom_config)
  200. return DictAttribute(usercfg)
  201. return {}
  202. def autodiscover_tasks(self, packages, related_name='tasks'):
  203. self.task_modules.update(
  204. mod.__name__ for mod in autodiscover_tasks(packages or (),
  205. related_name) if mod)
  206. @property
  207. def conf(self):
  208. """Loader configuration."""
  209. if self._conf is None:
  210. self._conf = self.read_configuration()
  211. return self._conf
  212. @cached_property
  213. def mail(self):
  214. return self.import_module('celery.utils.mail')
  215. def autodiscover_tasks(packages, related_name='tasks'):
  216. global _RACE_PROTECTION
  217. if _RACE_PROTECTION:
  218. return ()
  219. _RACE_PROTECTION = True
  220. try:
  221. return [find_related_module(pkg, related_name) for pkg in packages]
  222. finally:
  223. _RACE_PROTECTION = False
  224. def find_related_module(package, related_name):
  225. """Given a package name and a module name, tries to find that
  226. module."""
  227. # Django 1.7 allows for speciying a class name in INSTALLED_APPS.
  228. # (Issue #2248).
  229. try:
  230. importlib.import_module(package)
  231. except ImportError:
  232. package, _, _ = package.rpartition('.')
  233. try:
  234. pkg_path = importlib.import_module(package).__path__
  235. except AttributeError:
  236. return
  237. try:
  238. _imp.find_module(related_name, pkg_path)
  239. except ImportError:
  240. return
  241. return importlib.import_module('{0}.{1}'.format(package, related_name))