imports.py 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. # -*- coding: utf-8 -*-
  2. """
  3. celery.utils.import
  4. ~~~~~~~~~~~~~~~~~~~
  5. Utilities related to importing modules and symbols by name.
  6. """
  7. from __future__ import absolute_import
  8. import imp as _imp
  9. import importlib
  10. import os
  11. import sys
  12. from contextlib import contextmanager
  13. from kombu.utils import symbol_by_name
  14. from .compat import reload
  15. class NotAPackage(Exception):
  16. pass
  17. if sys.version_info >= (3, 3): # pragma: no cover
  18. def qualname(obj):
  19. return obj.__qualname__
  20. else:
  21. def qualname(obj): # noqa
  22. if not hasattr(obj, '__name__') and hasattr(obj, '__class__'):
  23. return qualname(obj.__class__)
  24. return '%s.%s' % (obj.__module__, obj.__name__)
  25. def instantiate(name, *args, **kwargs):
  26. """Instantiate class by name.
  27. See :func:`symbol_by_name`.
  28. """
  29. return symbol_by_name(name)(*args, **kwargs)
  30. @contextmanager
  31. def cwd_in_path():
  32. cwd = os.getcwd()
  33. if cwd in sys.path:
  34. yield
  35. else:
  36. sys.path.insert(0, cwd)
  37. try:
  38. yield cwd
  39. finally:
  40. try:
  41. sys.path.remove(cwd)
  42. except ValueError: # pragma: no cover
  43. pass
  44. def find_module(module, path=None, imp=None):
  45. """Version of :func:`imp.find_module` supporting dots."""
  46. if imp is None:
  47. imp = importlib.import_module
  48. with cwd_in_path():
  49. if '.' in module:
  50. last = None
  51. parts = module.split('.')
  52. for i, part in enumerate(parts[:-1]):
  53. mpart = imp('.'.join(parts[:i + 1]))
  54. try:
  55. path = mpart.__path__
  56. except AttributeError:
  57. raise NotAPackage(module)
  58. last = _imp.find_module(parts[i + 1], path)
  59. return last
  60. return _imp.find_module(module)
  61. def import_from_cwd(module, imp=None, package=None):
  62. """Import module, but make sure it finds modules
  63. located in the current directory.
  64. Modules located in the current directory has
  65. precedence over modules located in `sys.path`.
  66. """
  67. if imp is None:
  68. imp = importlib.import_module
  69. with cwd_in_path():
  70. return imp(module, package=package)
  71. def reload_from_cwd(module, reloader=None):
  72. if reloader is None:
  73. reloader = reload
  74. with cwd_in_path():
  75. return reloader(module)
  76. def module_file(module):
  77. name = module.__file__
  78. return name[:-1] if name.endswith('.pyc') else name