2015-03-11 16:18:42 +03:00
|
|
|
"""Utils module."""
|
|
|
|
|
2016-03-01 16:42:06 +03:00
|
|
|
import sys
|
2016-04-10 17:14:11 +03:00
|
|
|
import copy
|
|
|
|
import types
|
2015-09-04 02:33:15 +03:00
|
|
|
import threading
|
|
|
|
|
2015-09-01 00:36:26 +03:00
|
|
|
import six
|
2015-03-12 13:01:56 +03:00
|
|
|
|
2016-04-03 17:27:53 +03:00
|
|
|
from dependency_injector.errors import Error
|
2015-03-13 18:46:03 +03:00
|
|
|
|
2015-03-11 16:18:42 +03:00
|
|
|
|
2015-09-04 02:33:15 +03:00
|
|
|
GLOBAL_LOCK = threading.RLock()
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Dependency injector global reentrant lock.
|
|
|
|
|
|
|
|
:type: :py:class:`threading.RLock`
|
|
|
|
"""
|
2015-09-04 02:33:15 +03:00
|
|
|
|
2016-03-01 16:42:06 +03:00
|
|
|
_IS_PYPY = '__pypy__' in sys.builtin_module_names
|
|
|
|
if _IS_PYPY or six.PY3: # pragma: no cover
|
|
|
|
_OBJECT_INIT = six.get_unbound_function(object.__init__)
|
|
|
|
else: # pragma: no cover
|
|
|
|
_OBJECT_INIT = None
|
|
|
|
|
2016-04-10 17:14:11 +03:00
|
|
|
if six.PY2: # pragma: no cover
|
|
|
|
copy._deepcopy_dispatch[types.MethodType] = \
|
|
|
|
lambda obj, memo: type(obj)(obj.im_func,
|
|
|
|
copy.deepcopy(obj.im_self, memo),
|
|
|
|
obj.im_class)
|
|
|
|
|
2015-09-04 02:33:15 +03:00
|
|
|
|
2015-03-11 16:18:42 +03:00
|
|
|
def is_provider(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is provider instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2015-09-01 00:36:26 +03:00
|
|
|
return (not isinstance(instance, six.class_types) and
|
2015-10-23 18:07:52 +03:00
|
|
|
hasattr(instance, '__IS_PROVIDER__') and
|
|
|
|
getattr(instance, '__IS_PROVIDER__') is True)
|
2015-03-11 16:18:42 +03:00
|
|
|
|
|
|
|
|
2015-03-13 18:46:03 +03:00
|
|
|
def ensure_is_provider(instance):
|
2015-10-19 12:12:38 +03:00
|
|
|
"""Check if instance is provider instance and return it.
|
|
|
|
|
2015-11-25 15:41:03 +03:00
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:raise: :py:exc:`dependency_injector.errors.Error` if provided instance is
|
|
|
|
not provider.
|
|
|
|
|
|
|
|
:rtype: :py:class:`dependency_injector.providers.Provider`
|
2015-10-19 12:12:38 +03:00
|
|
|
"""
|
2015-03-13 18:46:03 +03:00
|
|
|
if not is_provider(instance):
|
|
|
|
raise Error('Expected provider instance, '
|
2015-03-18 13:48:19 +03:00
|
|
|
'got {0}'.format(str(instance)))
|
2015-03-13 18:46:03 +03:00
|
|
|
return instance
|
|
|
|
|
|
|
|
|
2015-12-28 18:25:25 +03:00
|
|
|
def is_delegated_provider(instance):
|
|
|
|
"""Check if instance is delegated provider instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
|
|
|
return (is_provider(instance) and
|
|
|
|
hasattr(instance, '__IS_DELEGATED__') and
|
|
|
|
getattr(instance, '__IS_DELEGATED__') is True)
|
|
|
|
|
|
|
|
|
2015-03-11 16:18:42 +03:00
|
|
|
def is_injection(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is injection instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2015-09-01 00:36:26 +03:00
|
|
|
return (not isinstance(instance, six.class_types) and
|
2015-10-23 18:07:52 +03:00
|
|
|
hasattr(instance, '__IS_INJECTION__') and
|
|
|
|
getattr(instance, '__IS_INJECTION__') is True)
|
2015-03-11 16:18:42 +03:00
|
|
|
|
|
|
|
|
2015-03-23 02:04:18 +03:00
|
|
|
def ensure_is_injection(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is injection instance and return it.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:raise: :py:exc:`dependency_injector.errors.Error` if provided instance is
|
|
|
|
not injection.
|
|
|
|
|
|
|
|
:rtype: :py:class:`dependency_injector.injections.Injection`
|
|
|
|
"""
|
2015-03-23 02:04:18 +03:00
|
|
|
if not is_injection(instance):
|
|
|
|
raise Error('Expected injection instance, '
|
|
|
|
'got {0}'.format(str(instance)))
|
|
|
|
return instance
|
|
|
|
|
|
|
|
|
2015-10-14 14:30:01 +03:00
|
|
|
def is_arg_injection(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is positional argument injection instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2015-10-14 14:30:01 +03:00
|
|
|
return (not isinstance(instance, six.class_types) and
|
2015-10-23 18:07:52 +03:00
|
|
|
hasattr(instance, '__IS_ARG_INJECTION__') and
|
2015-10-14 14:30:01 +03:00
|
|
|
getattr(instance, '__IS_ARG_INJECTION__', False) is True)
|
|
|
|
|
|
|
|
|
2015-03-23 02:04:18 +03:00
|
|
|
def is_kwarg_injection(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is keyword argument injection instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2015-09-01 00:36:26 +03:00
|
|
|
return (not isinstance(instance, six.class_types) and
|
2015-10-23 18:07:52 +03:00
|
|
|
hasattr(instance, '__IS_KWARG_INJECTION__') and
|
2015-07-22 10:53:16 +03:00
|
|
|
getattr(instance, '__IS_KWARG_INJECTION__', False) is True)
|
2015-03-11 16:18:42 +03:00
|
|
|
|
|
|
|
|
|
|
|
def is_attribute_injection(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is attribute injection instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2015-09-01 00:36:26 +03:00
|
|
|
return (not isinstance(instance, six.class_types) and
|
2015-10-23 18:07:52 +03:00
|
|
|
hasattr(instance, '__IS_ATTRIBUTE_INJECTION__') and
|
2015-07-22 10:53:16 +03:00
|
|
|
getattr(instance, '__IS_ATTRIBUTE_INJECTION__', False) is True)
|
2015-03-11 16:18:42 +03:00
|
|
|
|
|
|
|
|
2015-10-07 13:36:28 +03:00
|
|
|
def is_catalog(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is catalog instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2015-11-10 20:38:18 +03:00
|
|
|
return (hasattr(instance, '__IS_CATALOG__') and
|
2015-10-07 13:36:28 +03:00
|
|
|
getattr(instance, '__IS_CATALOG__', False) is True)
|
|
|
|
|
|
|
|
|
2015-11-10 20:38:18 +03:00
|
|
|
def is_dynamic_catalog(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is dynamic catalog instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2015-11-10 20:38:18 +03:00
|
|
|
return (not isinstance(instance, six.class_types) and is_catalog(instance))
|
|
|
|
|
|
|
|
|
|
|
|
def is_declarative_catalog(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is declarative catalog instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2015-11-10 20:38:18 +03:00
|
|
|
return (isinstance(instance, six.class_types) and is_catalog(instance))
|
|
|
|
|
|
|
|
|
2015-10-19 12:12:38 +03:00
|
|
|
def is_catalog_bundle(instance):
|
2015-11-25 15:41:03 +03:00
|
|
|
"""Check if instance is catalog bundle instance.
|
|
|
|
|
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:rtype: bool
|
|
|
|
"""
|
2015-10-19 12:12:38 +03:00
|
|
|
return (not isinstance(instance, six.class_types) and
|
2015-10-23 18:07:52 +03:00
|
|
|
hasattr(instance, '__IS_CATALOG_BUNDLE__') and
|
2015-10-19 12:12:38 +03:00
|
|
|
getattr(instance, '__IS_CATALOG_BUNDLE__', False) is True)
|
|
|
|
|
|
|
|
|
|
|
|
def ensure_is_catalog_bundle(instance):
|
|
|
|
"""Check if instance is catalog bundle instance and return it.
|
|
|
|
|
2015-11-25 15:41:03 +03:00
|
|
|
:param instance: Instance to be checked.
|
|
|
|
:type instance: object
|
|
|
|
|
|
|
|
:raise: :py:exc:`dependency_injector.errors.Error` if provided instance
|
|
|
|
is not catalog bundle.
|
|
|
|
|
|
|
|
:rtype: :py:class:`dependency_injector.catalogs.CatalogBundle`
|
2015-10-19 12:12:38 +03:00
|
|
|
"""
|
|
|
|
if not is_catalog_bundle(instance):
|
|
|
|
raise Error('Expected catalog bundle instance, '
|
|
|
|
'got {0}'.format(str(instance)))
|
|
|
|
return instance
|
2015-12-11 12:18:09 +03:00
|
|
|
|
|
|
|
|
|
|
|
def represent_provider(provider, provides):
|
|
|
|
"""Return string representation of provider.
|
|
|
|
|
|
|
|
:param provider: Provider object
|
|
|
|
:type provider: :py:class:`dependency_injector.providers.Provider`
|
|
|
|
|
|
|
|
:param provides: Object that provider provides
|
|
|
|
:type provider: object
|
|
|
|
|
|
|
|
:return: String representation of provider
|
|
|
|
:rtype: str
|
|
|
|
"""
|
|
|
|
return '<{provider}({provides}) at {address}>'.format(
|
|
|
|
provider='.'.join((provider.__class__.__module__,
|
|
|
|
provider.__class__.__name__)),
|
|
|
|
provides=repr(provides) if provides is not None else '',
|
|
|
|
address=hex(id(provider)))
|
2016-03-01 16:42:06 +03:00
|
|
|
|
|
|
|
|
|
|
|
def fetch_cls_init(cls):
|
|
|
|
"""Return reference to the class.__init__() method if it is defined.
|
|
|
|
|
|
|
|
:param cls: Class instance
|
|
|
|
:type cls: type
|
|
|
|
|
|
|
|
:return: Reference to the class.__init__() if any, or None otherwise.
|
|
|
|
:rtype: unbound method | None
|
|
|
|
"""
|
|
|
|
try:
|
|
|
|
cls_init = six.get_unbound_function(cls.__init__)
|
|
|
|
assert cls_init is not _OBJECT_INIT
|
|
|
|
except (AttributeError, AssertionError):
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
return cls_init
|
2016-04-10 17:14:11 +03:00
|
|
|
|
|
|
|
|
|
|
|
def _copy_providers(providers, memo=None):
|
|
|
|
"""Make full copy of providers dictionary."""
|
|
|
|
return copy.deepcopy(providers, memo)
|