mirror of
https://github.com/ets-labs/python-dependency-injector.git
synced 2024-11-25 11:04:01 +03:00
Add abstract providers
This commit is contained in:
parent
1dacd096f6
commit
2aa85228d7
|
@ -13,6 +13,9 @@ Development version
|
||||||
|
|
||||||
3.4.0
|
3.4.0
|
||||||
-----
|
-----
|
||||||
|
- Add ``AbstractCallable`` provider.
|
||||||
|
- Add ``AbstractFactory`` provider.
|
||||||
|
- Add ``AbstractSingleton`` provider.
|
||||||
- Optimize calling of overridden providers (~15% faster).
|
- Optimize calling of overridden providers (~15% faster).
|
||||||
|
|
||||||
3.3.7
|
3.3.7
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
|
@ -50,6 +50,10 @@ cdef class DelegatedCallable(Callable):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
cdef class AbstractCallable(Callable):
|
||||||
|
cpdef object _provide(self, tuple args, dict kwargs)
|
||||||
|
|
||||||
|
|
||||||
# Configuration providers
|
# Configuration providers
|
||||||
cdef class Configuration(Provider):
|
cdef class Configuration(Provider):
|
||||||
cdef str __name
|
cdef str __name
|
||||||
|
@ -76,6 +80,10 @@ cdef class DelegatedFactory(Factory):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
cdef class AbstractFactory(Factory):
|
||||||
|
cpdef object _provide(self, tuple args, dict kwargs)
|
||||||
|
|
||||||
|
|
||||||
# Singleton providers
|
# Singleton providers
|
||||||
cdef class BaseSingleton(Provider):
|
cdef class BaseSingleton(Provider):
|
||||||
cdef Factory __instantiator
|
cdef Factory __instantiator
|
||||||
|
@ -112,6 +120,10 @@ cdef class DelegatedThreadLocalSingleton(ThreadLocalSingleton):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
cdef class AbstractSingleton(BaseSingleton):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
# Injections
|
# Injections
|
||||||
cdef class Injection(object):
|
cdef class Injection(object):
|
||||||
cdef object __value
|
cdef object __value
|
||||||
|
|
|
@ -593,6 +593,47 @@ cdef class DelegatedCallable(Callable):
|
||||||
__IS_DELEGATED__ = True
|
__IS_DELEGATED__ = True
|
||||||
|
|
||||||
|
|
||||||
|
cdef class AbstractCallable(Callable):
|
||||||
|
"""Abstract callable provider.
|
||||||
|
|
||||||
|
:py:class:`AbstractCallable` is a :py:class:`Callable` provider that must
|
||||||
|
be explicitly overridden before calling.
|
||||||
|
|
||||||
|
Overriding of :py:class:`AbstractCallable` is possible only by another
|
||||||
|
:py:class:`Callable` provider.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __call__(self, *args, **kwargs):
|
||||||
|
"""Return provided object.
|
||||||
|
|
||||||
|
Callable interface implementation.
|
||||||
|
"""
|
||||||
|
if self.__last_overriding is None:
|
||||||
|
raise Error('{0} must be overridden before calling'.format(self))
|
||||||
|
return self.__last_overriding._provide(args, kwargs)
|
||||||
|
|
||||||
|
def override(self, provider):
|
||||||
|
"""Override provider with another provider.
|
||||||
|
|
||||||
|
:param provider: Overriding provider.
|
||||||
|
:type provider: :py:class:`Provider`
|
||||||
|
|
||||||
|
:raise: :py:exc:`dependency_injector.errors.Error`
|
||||||
|
|
||||||
|
:return: Overriding context.
|
||||||
|
:rtype: :py:class:`OverridingContext`
|
||||||
|
"""
|
||||||
|
if not isinstance(provider, Callable):
|
||||||
|
raise Error('{0} must be overridden only by '
|
||||||
|
'{1} providers'.format(self, Callable))
|
||||||
|
return super(AbstractCallable, self).override(provider)
|
||||||
|
|
||||||
|
cpdef object _provide(self, tuple args, dict kwargs):
|
||||||
|
"""Return result of provided callable's call."""
|
||||||
|
raise NotImplementedError('Abstract provider forward providing logic '
|
||||||
|
'to overriding provider')
|
||||||
|
|
||||||
|
|
||||||
cdef class Configuration(Provider):
|
cdef class Configuration(Provider):
|
||||||
"""Configuration provider.
|
"""Configuration provider.
|
||||||
|
|
||||||
|
@ -967,6 +1008,46 @@ cdef class DelegatedFactory(Factory):
|
||||||
__IS_DELEGATED__ = True
|
__IS_DELEGATED__ = True
|
||||||
|
|
||||||
|
|
||||||
|
cdef class AbstractFactory(Factory):
|
||||||
|
"""Abstract factory provider.
|
||||||
|
|
||||||
|
:py:class:`AbstractFactory` is a :py:class:`Factory` provider that must
|
||||||
|
be explicitly overridden before calling.
|
||||||
|
|
||||||
|
Overriding of :py:class:`AbstractFactory` is possible only by another
|
||||||
|
:py:class:`Factory` provider.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __call__(self, *args, **kwargs):
|
||||||
|
"""Return provided object.
|
||||||
|
|
||||||
|
Callable interface implementation.
|
||||||
|
"""
|
||||||
|
if self.__last_overriding is None:
|
||||||
|
raise Error('{0} must be overridden before calling'.format(self))
|
||||||
|
return self.__last_overriding._provide(args, kwargs)
|
||||||
|
|
||||||
|
def override(self, provider):
|
||||||
|
"""Override provider with another provider.
|
||||||
|
|
||||||
|
:param provider: Overriding provider.
|
||||||
|
:type provider: :py:class:`Provider`
|
||||||
|
|
||||||
|
:raise: :py:exc:`dependency_injector.errors.Error`
|
||||||
|
|
||||||
|
:return: Overriding context.
|
||||||
|
:rtype: :py:class:`OverridingContext`
|
||||||
|
"""
|
||||||
|
if not isinstance(provider, Factory):
|
||||||
|
raise Error('{0} must be overridden only by '
|
||||||
|
'{1} providers'.format(self, Factory))
|
||||||
|
return super(AbstractFactory, self).override(provider)
|
||||||
|
|
||||||
|
cpdef object _provide(self, tuple args, dict kwargs):
|
||||||
|
"""Return result of provided callable's call."""
|
||||||
|
raise NotImplementedError('Abstract provider forward providing logic '
|
||||||
|
'to overriding provider')
|
||||||
|
|
||||||
cdef class BaseSingleton(Provider):
|
cdef class BaseSingleton(Provider):
|
||||||
"""Base class of singleton providers."""
|
"""Base class of singleton providers."""
|
||||||
|
|
||||||
|
@ -1353,6 +1434,51 @@ cdef class DelegatedThreadLocalSingleton(ThreadLocalSingleton):
|
||||||
__IS_DELEGATED__ = True
|
__IS_DELEGATED__ = True
|
||||||
|
|
||||||
|
|
||||||
|
cdef class AbstractSingleton(BaseSingleton):
|
||||||
|
"""Abstract singleton provider.
|
||||||
|
|
||||||
|
:py:class:`AbstractSingleton` is a :py:class:`Singleton` provider that must
|
||||||
|
be explicitly overridden before calling.
|
||||||
|
|
||||||
|
Overriding of :py:class:`AbstractSingleton` is possible only by another
|
||||||
|
:py:class:`BaseSingleton` provider.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __call__(self, *args, **kwargs):
|
||||||
|
"""Return provided object.
|
||||||
|
|
||||||
|
Callable interface implementation.
|
||||||
|
"""
|
||||||
|
if self.__last_overriding is None:
|
||||||
|
raise Error('{0} must be overridden before calling'.format(self))
|
||||||
|
return self.__last_overriding._provide(args, kwargs)
|
||||||
|
|
||||||
|
def override(self, provider):
|
||||||
|
"""Override provider with another provider.
|
||||||
|
|
||||||
|
:param provider: Overriding provider.
|
||||||
|
:type provider: :py:class:`Provider`
|
||||||
|
|
||||||
|
:raise: :py:exc:`dependency_injector.errors.Error`
|
||||||
|
|
||||||
|
:return: Overriding context.
|
||||||
|
:rtype: :py:class:`OverridingContext`
|
||||||
|
"""
|
||||||
|
if not isinstance(provider, BaseSingleton):
|
||||||
|
raise Error('{0} must be overridden only by '
|
||||||
|
'{1} providers'.format(self, BaseSingleton))
|
||||||
|
return super(AbstractSingleton, self).override(provider)
|
||||||
|
|
||||||
|
def reset(self):
|
||||||
|
"""Reset cached instance, if any.
|
||||||
|
|
||||||
|
:rtype: None
|
||||||
|
"""
|
||||||
|
if self.__last_overriding is None:
|
||||||
|
raise Error('{0} must be overridden before calling'.format(self))
|
||||||
|
return self.__last_overriding.reset()
|
||||||
|
|
||||||
|
|
||||||
cdef class Injection(object):
|
cdef class Injection(object):
|
||||||
"""Abstract injection class."""
|
"""Abstract injection class."""
|
||||||
|
|
||||||
|
|
|
@ -117,7 +117,7 @@ class Tester(object):
|
||||||
for x in xrange(int(5000000 * self.duration_factor)):
|
for x in xrange(int(5000000 * self.duration_factor)):
|
||||||
test_factory()
|
test_factory()
|
||||||
|
|
||||||
def test_overridden_factory_3_factory_kw_injections(self, providers):
|
def test_abstract_factory_3_factory_kw_injections(self, providers):
|
||||||
"""Test factory with 3 keyword argument injections via factories."""
|
"""Test factory with 3 keyword argument injections via factories."""
|
||||||
class A(object):
|
class A(object):
|
||||||
pass
|
pass
|
||||||
|
@ -135,7 +135,7 @@ class Tester(object):
|
||||||
a_factory = providers.Factory(A)
|
a_factory = providers.Factory(A)
|
||||||
b_factory = providers.Factory(B)
|
b_factory = providers.Factory(B)
|
||||||
c_factory = providers.Factory(C)
|
c_factory = providers.Factory(C)
|
||||||
test_factory = providers.Factory(object)
|
test_factory = providers.AbstractFactory(object)
|
||||||
test_factory.override(providers.Factory(Test,
|
test_factory.override(providers.Factory(Test,
|
||||||
a=a_factory,
|
a=a_factory,
|
||||||
b=b_factory,
|
b=b_factory,
|
||||||
|
|
|
@ -8,13 +8,14 @@ from dependency_injector import (
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class CallableTests(unittest.TestCase):
|
def _example(arg1, arg2, arg3, arg4):
|
||||||
|
|
||||||
def example(self, arg1, arg2, arg3, arg4):
|
|
||||||
return arg1, arg2, arg3, arg4
|
return arg1, arg2, arg3, arg4
|
||||||
|
|
||||||
|
|
||||||
|
class CallableTests(unittest.TestCase):
|
||||||
|
|
||||||
def test_init_with_callable(self):
|
def test_init_with_callable(self):
|
||||||
self.assertTrue(providers.Callable(self.example))
|
self.assertTrue(providers.Callable(_example))
|
||||||
|
|
||||||
def test_init_with_not_callable(self):
|
def test_init_with_not_callable(self):
|
||||||
self.assertRaises(errors.Error, providers.Callable, 123)
|
self.assertRaises(errors.Error, providers.Callable, 123)
|
||||||
|
@ -24,66 +25,66 @@ class CallableTests(unittest.TestCase):
|
||||||
self.assertTrue(provider())
|
self.assertTrue(provider())
|
||||||
|
|
||||||
def test_call_with_positional_args(self):
|
def test_call_with_positional_args(self):
|
||||||
provider = providers.Callable(self.example,
|
provider = providers.Callable(_example,
|
||||||
1, 2, 3, 4)
|
1, 2, 3, 4)
|
||||||
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
||||||
|
|
||||||
def test_call_with_keyword_args(self):
|
def test_call_with_keyword_args(self):
|
||||||
provider = providers.Callable(self.example,
|
provider = providers.Callable(_example,
|
||||||
arg1=1, arg2=2, arg3=3, arg4=4)
|
arg1=1, arg2=2, arg3=3, arg4=4)
|
||||||
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
||||||
|
|
||||||
def test_call_with_positional_and_keyword_args(self):
|
def test_call_with_positional_and_keyword_args(self):
|
||||||
provider = providers.Callable(self.example,
|
provider = providers.Callable(_example,
|
||||||
1, 2,
|
1, 2,
|
||||||
arg3=3, arg4=4)
|
arg3=3, arg4=4)
|
||||||
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
||||||
|
|
||||||
def test_call_with_context_args(self):
|
def test_call_with_context_args(self):
|
||||||
provider = providers.Callable(self.example, 1, 2)
|
provider = providers.Callable(_example, 1, 2)
|
||||||
self.assertTupleEqual(provider(3, 4), (1, 2, 3, 4))
|
self.assertTupleEqual(provider(3, 4), (1, 2, 3, 4))
|
||||||
|
|
||||||
def test_call_with_context_kwargs(self):
|
def test_call_with_context_kwargs(self):
|
||||||
provider = providers.Callable(self.example, arg1=1)
|
provider = providers.Callable(_example, arg1=1)
|
||||||
self.assertTupleEqual(provider(arg2=2, arg3=3, arg4=4), (1, 2, 3, 4))
|
self.assertTupleEqual(provider(arg2=2, arg3=3, arg4=4), (1, 2, 3, 4))
|
||||||
|
|
||||||
def test_call_with_context_args_and_kwargs(self):
|
def test_call_with_context_args_and_kwargs(self):
|
||||||
provider = providers.Callable(self.example, 1)
|
provider = providers.Callable(_example, 1)
|
||||||
self.assertTupleEqual(provider(2, arg3=3, arg4=4), (1, 2, 3, 4))
|
self.assertTupleEqual(provider(2, arg3=3, arg4=4), (1, 2, 3, 4))
|
||||||
|
|
||||||
def test_fluent_interface(self):
|
def test_fluent_interface(self):
|
||||||
provider = providers.Singleton(self.example) \
|
provider = providers.Singleton(_example) \
|
||||||
.add_args(1, 2) \
|
.add_args(1, 2) \
|
||||||
.add_kwargs(arg3=3, arg4=4)
|
.add_kwargs(arg3=3, arg4=4)
|
||||||
|
|
||||||
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
||||||
|
|
||||||
def test_set_args(self):
|
def test_set_args(self):
|
||||||
provider = providers.Callable(self.example) \
|
provider = providers.Callable(_example) \
|
||||||
.add_args(1, 2) \
|
.add_args(1, 2) \
|
||||||
.set_args(3, 4)
|
.set_args(3, 4)
|
||||||
self.assertEqual(provider.args, tuple([3, 4]))
|
self.assertEqual(provider.args, tuple([3, 4]))
|
||||||
|
|
||||||
def test_set_kwargs(self):
|
def test_set_kwargs(self):
|
||||||
provider = providers.Callable(self.example) \
|
provider = providers.Callable(_example) \
|
||||||
.add_kwargs(init_arg3=3, init_arg4=4) \
|
.add_kwargs(init_arg3=3, init_arg4=4) \
|
||||||
.set_kwargs(init_arg3=4, init_arg4=5)
|
.set_kwargs(init_arg3=4, init_arg4=5)
|
||||||
self.assertEqual(provider.kwargs, dict(init_arg3=4, init_arg4=5))
|
self.assertEqual(provider.kwargs, dict(init_arg3=4, init_arg4=5))
|
||||||
|
|
||||||
def test_clear_args(self):
|
def test_clear_args(self):
|
||||||
provider = providers.Callable(self.example) \
|
provider = providers.Callable(_example) \
|
||||||
.add_args(1, 2) \
|
.add_args(1, 2) \
|
||||||
.clear_args()
|
.clear_args()
|
||||||
self.assertEqual(provider.args, tuple())
|
self.assertEqual(provider.args, tuple())
|
||||||
|
|
||||||
def test_clear_kwargs(self):
|
def test_clear_kwargs(self):
|
||||||
provider = providers.Callable(self.example) \
|
provider = providers.Callable(_example) \
|
||||||
.add_kwargs(init_arg3=3, init_arg4=4) \
|
.add_kwargs(init_arg3=3, init_arg4=4) \
|
||||||
.clear_kwargs()
|
.clear_kwargs()
|
||||||
self.assertEqual(provider.kwargs, dict())
|
self.assertEqual(provider.kwargs, dict())
|
||||||
|
|
||||||
def test_call_overridden(self):
|
def test_call_overridden(self):
|
||||||
provider = providers.Callable(self.example)
|
provider = providers.Callable(_example)
|
||||||
|
|
||||||
provider.override(providers.Object((4, 3, 2, 1)))
|
provider.override(providers.Object((4, 3, 2, 1)))
|
||||||
provider.override(providers.Object((1, 2, 3, 4)))
|
provider.override(providers.Object((1, 2, 3, 4)))
|
||||||
|
@ -91,7 +92,7 @@ class CallableTests(unittest.TestCase):
|
||||||
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
self.assertTupleEqual(provider(), (1, 2, 3, 4))
|
||||||
|
|
||||||
def test_deepcopy(self):
|
def test_deepcopy(self):
|
||||||
provider = providers.Callable(self.example)
|
provider = providers.Callable(_example)
|
||||||
|
|
||||||
provider_copy = providers.deepcopy(provider)
|
provider_copy = providers.deepcopy(provider)
|
||||||
|
|
||||||
|
@ -100,8 +101,8 @@ class CallableTests(unittest.TestCase):
|
||||||
self.assertIsInstance(provider, providers.Callable)
|
self.assertIsInstance(provider, providers.Callable)
|
||||||
|
|
||||||
def test_deepcopy_from_memo(self):
|
def test_deepcopy_from_memo(self):
|
||||||
provider = providers.Callable(self.example)
|
provider = providers.Callable(_example)
|
||||||
provider_copy_memo = providers.Callable(self.example)
|
provider_copy_memo = providers.Callable(_example)
|
||||||
|
|
||||||
provider_copy = providers.deepcopy(
|
provider_copy = providers.deepcopy(
|
||||||
provider, memo={id(provider): provider_copy_memo})
|
provider, memo={id(provider): provider_copy_memo})
|
||||||
|
@ -109,7 +110,7 @@ class CallableTests(unittest.TestCase):
|
||||||
self.assertIs(provider_copy, provider_copy_memo)
|
self.assertIs(provider_copy, provider_copy_memo)
|
||||||
|
|
||||||
def test_deepcopy_args(self):
|
def test_deepcopy_args(self):
|
||||||
provider = providers.Callable(self.example)
|
provider = providers.Callable(_example)
|
||||||
dependent_provider1 = providers.Callable(list)
|
dependent_provider1 = providers.Callable(list)
|
||||||
dependent_provider2 = providers.Callable(dict)
|
dependent_provider2 = providers.Callable(dict)
|
||||||
|
|
||||||
|
@ -130,7 +131,7 @@ class CallableTests(unittest.TestCase):
|
||||||
self.assertIsNot(dependent_provider2, dependent_provider_copy2)
|
self.assertIsNot(dependent_provider2, dependent_provider_copy2)
|
||||||
|
|
||||||
def test_deepcopy_kwargs(self):
|
def test_deepcopy_kwargs(self):
|
||||||
provider = providers.Callable(self.example)
|
provider = providers.Callable(_example)
|
||||||
dependent_provider1 = providers.Callable(list)
|
dependent_provider1 = providers.Callable(list)
|
||||||
dependent_provider2 = providers.Callable(dict)
|
dependent_provider2 = providers.Callable(dict)
|
||||||
|
|
||||||
|
@ -151,7 +152,7 @@ class CallableTests(unittest.TestCase):
|
||||||
self.assertIsNot(dependent_provider2, dependent_provider_copy2)
|
self.assertIsNot(dependent_provider2, dependent_provider_copy2)
|
||||||
|
|
||||||
def test_deepcopy_overridden(self):
|
def test_deepcopy_overridden(self):
|
||||||
provider = providers.Callable(self.example)
|
provider = providers.Callable(_example)
|
||||||
object_provider = providers.Object(object())
|
object_provider = providers.Object(object())
|
||||||
|
|
||||||
provider.override(object_provider)
|
provider.override(object_provider)
|
||||||
|
@ -167,34 +168,86 @@ class CallableTests(unittest.TestCase):
|
||||||
self.assertIsInstance(object_provider_copy, providers.Object)
|
self.assertIsInstance(object_provider_copy, providers.Object)
|
||||||
|
|
||||||
def test_repr(self):
|
def test_repr(self):
|
||||||
provider = providers.Callable(self.example)
|
provider = providers.Callable(_example)
|
||||||
|
|
||||||
self.assertEqual(repr(provider),
|
self.assertEqual(repr(provider),
|
||||||
'<dependency_injector.providers.'
|
'<dependency_injector.providers.'
|
||||||
'Callable({0}) at {1}>'.format(
|
'Callable({0}) at {1}>'.format(
|
||||||
repr(self.example),
|
repr(_example),
|
||||||
hex(id(provider))))
|
hex(id(provider))))
|
||||||
|
|
||||||
|
|
||||||
class DelegatedCallableTests(unittest.TestCase):
|
class DelegatedCallableTests(unittest.TestCase):
|
||||||
|
|
||||||
def test_inheritance(self):
|
def test_inheritance(self):
|
||||||
self.assertIsInstance(providers.DelegatedCallable(len),
|
self.assertIsInstance(providers.DelegatedCallable(_example),
|
||||||
providers.Callable)
|
providers.Callable)
|
||||||
|
|
||||||
def test_is_provider(self):
|
def test_is_provider(self):
|
||||||
self.assertTrue(
|
self.assertTrue(
|
||||||
providers.is_provider(providers.DelegatedCallable(len)))
|
providers.is_provider(providers.DelegatedCallable(_example)))
|
||||||
|
|
||||||
def test_is_delegated_provider(self):
|
def test_is_delegated_provider(self):
|
||||||
provider = providers.DelegatedCallable(len)
|
provider = providers.DelegatedCallable(_example)
|
||||||
self.assertTrue(providers.is_delegated(provider))
|
self.assertTrue(providers.is_delegated(provider))
|
||||||
|
|
||||||
def test_repr(self):
|
def test_repr(self):
|
||||||
provider = providers.DelegatedCallable(len)
|
provider = providers.DelegatedCallable(_example)
|
||||||
|
|
||||||
self.assertEqual(repr(provider),
|
self.assertEqual(repr(provider),
|
||||||
'<dependency_injector.providers.'
|
'<dependency_injector.providers.'
|
||||||
'DelegatedCallable({0}) at {1}>'.format(
|
'DelegatedCallable({0}) at {1}>'.format(
|
||||||
repr(len),
|
repr(_example),
|
||||||
|
hex(id(provider))))
|
||||||
|
|
||||||
|
|
||||||
|
class AbstractCallableTests(unittest.TestCase):
|
||||||
|
|
||||||
|
def test_inheritance(self):
|
||||||
|
self.assertIsInstance(providers.AbstractCallable(_example),
|
||||||
|
providers.Callable)
|
||||||
|
|
||||||
|
def test_call_overridden_by_callable(self):
|
||||||
|
def _abstract_example():
|
||||||
|
pass
|
||||||
|
|
||||||
|
provider = providers.AbstractCallable(_abstract_example)
|
||||||
|
provider.override(providers.Callable(_example))
|
||||||
|
|
||||||
|
self.assertTrue(provider(1, 2, 3, 4), (1, 2, 3, 4))
|
||||||
|
|
||||||
|
def test_call_overridden_by_delegated_callable(self):
|
||||||
|
def _abstract_example():
|
||||||
|
pass
|
||||||
|
|
||||||
|
provider = providers.AbstractCallable(_abstract_example)
|
||||||
|
provider.override(providers.DelegatedCallable(_example))
|
||||||
|
|
||||||
|
self.assertTrue(provider(1, 2, 3, 4), (1, 2, 3, 4))
|
||||||
|
|
||||||
|
def test_call_not_overridden(self):
|
||||||
|
provider = providers.AbstractCallable(_example)
|
||||||
|
|
||||||
|
with self.assertRaises(errors.Error):
|
||||||
|
provider(1, 2, 3, 4)
|
||||||
|
|
||||||
|
def test_override_by_not_callable(self):
|
||||||
|
provider = providers.AbstractCallable(_example)
|
||||||
|
|
||||||
|
with self.assertRaises(errors.Error):
|
||||||
|
provider.override(providers.Factory(object))
|
||||||
|
|
||||||
|
def test_provide_not_implemented(self):
|
||||||
|
provider = providers.AbstractCallable(_example)
|
||||||
|
|
||||||
|
with self.assertRaises(NotImplementedError):
|
||||||
|
provider._provide((1, 2, 3, 4), dict())
|
||||||
|
|
||||||
|
def test_repr(self):
|
||||||
|
provider = providers.AbstractCallable(_example)
|
||||||
|
|
||||||
|
self.assertEqual(repr(provider),
|
||||||
|
'<dependency_injector.providers.'
|
||||||
|
'AbstractCallable({0}) at {1}>'.format(
|
||||||
|
repr(_example),
|
||||||
hex(id(provider))))
|
hex(id(provider))))
|
||||||
|
|
|
@ -353,3 +353,49 @@ class DelegatedFactoryTests(unittest.TestCase):
|
||||||
'DelegatedFactory({0}) at {1}>'.format(
|
'DelegatedFactory({0}) at {1}>'.format(
|
||||||
repr(Example),
|
repr(Example),
|
||||||
hex(id(provider))))
|
hex(id(provider))))
|
||||||
|
|
||||||
|
|
||||||
|
class AbstractFactoryTests(unittest.TestCase):
|
||||||
|
|
||||||
|
def test_inheritance(self):
|
||||||
|
self.assertIsInstance(providers.AbstractFactory(Example),
|
||||||
|
providers.Factory)
|
||||||
|
|
||||||
|
def test_call_overridden_by_factory(self):
|
||||||
|
provider = providers.AbstractFactory(object)
|
||||||
|
provider.override(providers.Factory(Example))
|
||||||
|
|
||||||
|
self.assertIsInstance(provider(), Example)
|
||||||
|
|
||||||
|
def test_call_overridden_by_delegated_factory(self):
|
||||||
|
provider = providers.AbstractFactory(object)
|
||||||
|
provider.override(providers.DelegatedFactory(Example))
|
||||||
|
|
||||||
|
self.assertIsInstance(provider(), Example)
|
||||||
|
|
||||||
|
def test_call_not_overridden(self):
|
||||||
|
provider = providers.AbstractFactory(object)
|
||||||
|
|
||||||
|
with self.assertRaises(errors.Error):
|
||||||
|
provider()
|
||||||
|
|
||||||
|
def test_override_by_not_factory(self):
|
||||||
|
provider = providers.AbstractFactory(object)
|
||||||
|
|
||||||
|
with self.assertRaises(errors.Error):
|
||||||
|
provider.override(providers.Callable(object))
|
||||||
|
|
||||||
|
def test_provide_not_implemented(self):
|
||||||
|
provider = providers.AbstractFactory(Example)
|
||||||
|
|
||||||
|
with self.assertRaises(NotImplementedError):
|
||||||
|
provider._provide(tuple(), dict())
|
||||||
|
|
||||||
|
def test_repr(self):
|
||||||
|
provider = providers.AbstractFactory(Example)
|
||||||
|
|
||||||
|
self.assertEqual(repr(provider),
|
||||||
|
'<dependency_injector.providers.'
|
||||||
|
'AbstractFactory({0}) at {1}>'.format(
|
||||||
|
repr(Example),
|
||||||
|
hex(id(provider))))
|
||||||
|
|
|
@ -432,3 +432,63 @@ class DelegatedThreadSafeSingletonTests(_BaseSingletonTestCase,
|
||||||
'DelegatedThreadSafeSingleton({0}) at {1}>'.format(
|
'DelegatedThreadSafeSingleton({0}) at {1}>'.format(
|
||||||
repr(Example),
|
repr(Example),
|
||||||
hex(id(provider))))
|
hex(id(provider))))
|
||||||
|
|
||||||
|
|
||||||
|
class AbstractSingletonTests(unittest.TestCase):
|
||||||
|
|
||||||
|
def test_inheritance(self):
|
||||||
|
self.assertIsInstance(providers.AbstractSingleton(Example),
|
||||||
|
providers.BaseSingleton)
|
||||||
|
|
||||||
|
def test_call_overridden_by_singleton(self):
|
||||||
|
provider = providers.AbstractSingleton(object)
|
||||||
|
provider.override(providers.Singleton(Example))
|
||||||
|
|
||||||
|
self.assertIsInstance(provider(), Example)
|
||||||
|
|
||||||
|
def test_call_overridden_by_delegated_singleton(self):
|
||||||
|
provider = providers.AbstractSingleton(object)
|
||||||
|
provider.override(providers.DelegatedSingleton(Example))
|
||||||
|
|
||||||
|
self.assertIsInstance(provider(), Example)
|
||||||
|
|
||||||
|
def test_call_not_overridden(self):
|
||||||
|
provider = providers.AbstractSingleton(object)
|
||||||
|
|
||||||
|
with self.assertRaises(errors.Error):
|
||||||
|
provider()
|
||||||
|
|
||||||
|
def test_reset_overridden(self):
|
||||||
|
provider = providers.AbstractSingleton(object)
|
||||||
|
provider.override(providers.Singleton(Example))
|
||||||
|
|
||||||
|
instance1 = provider()
|
||||||
|
|
||||||
|
provider.reset()
|
||||||
|
|
||||||
|
instance2 = provider()
|
||||||
|
|
||||||
|
self.assertIsNot(instance1, instance2)
|
||||||
|
self.assertIsInstance(instance1, Example)
|
||||||
|
self.assertIsInstance(instance2, Example)
|
||||||
|
|
||||||
|
def test_reset_not_overridden(self):
|
||||||
|
provider = providers.AbstractSingleton(object)
|
||||||
|
|
||||||
|
with self.assertRaises(errors.Error):
|
||||||
|
provider.reset()
|
||||||
|
|
||||||
|
def test_override_by_not_singleton(self):
|
||||||
|
provider = providers.AbstractSingleton(object)
|
||||||
|
|
||||||
|
with self.assertRaises(errors.Error):
|
||||||
|
provider.override(providers.Factory(object))
|
||||||
|
|
||||||
|
def test_repr(self):
|
||||||
|
provider = providers.AbstractSingleton(Example)
|
||||||
|
|
||||||
|
self.assertEqual(repr(provider),
|
||||||
|
'<dependency_injector.providers.'
|
||||||
|
'AbstractSingleton({0}) at {1}>'.format(
|
||||||
|
repr(Example),
|
||||||
|
hex(id(provider))))
|
||||||
|
|
Loading…
Reference in New Issue
Block a user