python-dependency-injector/dependency_injector/containers.py

105 lines
3.4 KiB
Python
Raw Normal View History

2016-05-27 14:37:15 +03:00
"""IoC containers module."""
import six
from dependency_injector import (
utils,
errors,
)
2016-05-27 14:37:15 +03:00
class DeclarativeContainerMetaClass(type):
"""Declarative inversion of control container meta class."""
def __new__(mcs, class_name, bases, attributes):
"""Declarative container class factory."""
cls_providers = tuple((name, provider)
for name, provider in six.iteritems(attributes)
if utils.is_provider(provider))
inherited_providers = tuple((name, provider)
for base in bases if utils.is_catalog(base)
for name, provider in six.iteritems(
base.cls_providers))
attributes['cls_providers'] = dict(cls_providers)
attributes['inherited_providers'] = dict(inherited_providers)
attributes['providers'] = dict(cls_providers + inherited_providers)
2016-05-27 14:37:15 +03:00
return type.__new__(mcs, class_name, bases, attributes)
def __setattr__(cls, name, value):
"""Set class attribute.
If value of attribute is provider, it will be added into providers
dictionary.
"""
if utils.is_provider(value):
cls.providers[name] = value
cls.cls_providers[name] = value
2016-05-27 14:37:15 +03:00
super(DeclarativeContainerMetaClass, cls).__setattr__(name, value)
def __delattr__(cls, name):
"""Delete class attribute.
2016-05-27 14:37:15 +03:00
If value of attribute is provider, it will be deleted from providers
2016-05-27 14:37:15 +03:00
dictionary.
"""
if name in cls.providers and name in cls.cls_providers:
del cls.providers[name]
del cls.cls_providers[name]
super(DeclarativeContainerMetaClass, cls).__delattr__(name)
2016-05-27 14:37:15 +03:00
@six.add_metaclass(DeclarativeContainerMetaClass)
class DeclarativeContainer(object):
"""Declarative inversion of control container."""
__IS_CATALOG__ = True
providers = dict()
2016-05-27 14:37:15 +03:00
cls_providers = dict()
inherited_providers = dict()
overridden_by = tuple()
@classmethod
def override(cls, overriding):
"""Override current container by overriding container.
:param overriding: Overriding container.
:type overriding: :py:class:`DeclarativeContainer`
:raise: :py:exc:`dependency_injector.errors.Error` if trying to
override container by itself or its subclasses
:rtype: None
"""
if issubclass(cls, overriding):
raise errors.Error('Catalog {0} could not be overridden '
'with itself or its subclasses'.format(cls))
cls.overridden_by += (overriding,)
for name, provider in six.iteritems(overriding.cls_providers):
try:
getattr(cls, name).override(provider)
except AttributeError:
pass
2016-05-27 14:37:15 +03:00
def override(container):
2016-05-27 14:37:15 +03:00
""":py:class:`DeclarativeContainer` overriding decorator.
:param catalog: Container that should be overridden by decorated container.
:type catalog: :py:class:`DeclarativeContainer`
2016-05-27 14:37:15 +03:00
:return: Declarative container's overriding decorator.
:rtype: callable(:py:class:`DeclarativeContainer`)
"""
def decorator(overriding_container):
"""Overriding decorator."""
container.override(overriding_container)
2016-05-27 14:37:15 +03:00
return overriding_container
return decorator