python-dependency-injector/dependency_injector/utils.py

114 lines
3.0 KiB
Python
Raw Normal View History

2016-06-03 12:03:27 +03:00
"""Dependency injector utils module."""
2016-03-01 16:42:06 +03:00
import sys
2016-06-01 18:53:35 +03:00
import copy as _copy
2016-04-10 17:14:11 +03:00
import types
2015-09-04 02:33:15 +03:00
import threading
2015-09-01 00:36:26 +03:00
import six
2016-05-29 16:39:39 +03:00
from dependency_injector import errors
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
2016-06-01 18:53:35 +03:00
_copy._deepcopy_dispatch[types.MethodType] = \
2016-04-10 17:14:11 +03:00
lambda obj, memo: type(obj)(obj.im_func,
2016-06-01 18:53:35 +03:00
_copy.deepcopy(obj.im_self, memo),
2016-04-10 17:14:11 +03:00
obj.im_class)
2015-09-04 02:33:15 +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
hasattr(instance, '__IS_PROVIDER__') and
getattr(instance, '__IS_PROVIDER__') is True)
def ensure_is_provider(instance):
"""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`
"""
if not is_provider(instance):
2016-05-29 16:39:39 +03:00
raise errors.Error('Expected provider instance, '
'got {0}'.format(str(instance)))
return instance
def is_container(instance):
"""Check if instance is container instance.
:param instance: Instance to be checked.
:type instance: object
:rtype: bool
"""
return (hasattr(instance, '__IS_CONTAINER__') and
getattr(instance, '__IS_CONTAINER__', False) is True)
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
2016-06-01 18:53:35 +03:00
def deepcopy(instance, memo=None):
"""Make full copy of instance."""
return _copy.deepcopy(instance, memo)