2015-08-31 16:31:38 +03:00
|
|
|
Dependency Injector
|
|
|
|
===================
|
2015-04-03 00:29:00 +03:00
|
|
|
|
2016-02-04 23:47:45 +03:00
|
|
|
*Dependency Injector* is a Python dependency injection framework. It was
|
|
|
|
designed to be unified, developer's friendly tool for managing any kind
|
|
|
|
of Python objects and their dependencies in formal, pretty way.
|
|
|
|
|
|
|
|
Below is a list of some key features and points of *Dependency Injector*:
|
|
|
|
|
|
|
|
- Easy, smart, pythonic style.
|
|
|
|
- Obvious, clear structure.
|
|
|
|
- Memory efficiency.
|
|
|
|
- Thread safety.
|
|
|
|
- Semantic versioning.
|
|
|
|
|
|
|
|
Main idea of *Dependency Injector* is to keep dependencies under control.
|
|
|
|
|
|
|
|
Status
|
|
|
|
------
|
2015-04-03 00:29:00 +03:00
|
|
|
|
2016-01-11 11:53:35 +03:00
|
|
|
+---------------------------------------+---------------------------------------------------------------------------------+
|
|
|
|
| *PyPi* | .. image:: https://img.shields.io/pypi/v/dependency_injector.svg |
|
|
|
|
| | :target: https://pypi.python.org/pypi/dependency_injector/ |
|
|
|
|
| | :alt: Latest Version |
|
|
|
|
| | .. image:: https://img.shields.io/pypi/dm/dependency_injector.svg |
|
|
|
|
| | :target: https://pypi.python.org/pypi/dependency_injector/ |
|
|
|
|
| | :alt: Downloads |
|
|
|
|
| | .. image:: https://img.shields.io/pypi/l/dependency_injector.svg |
|
|
|
|
| | :target: https://pypi.python.org/pypi/dependency_injector/ |
|
|
|
|
| | :alt: License |
|
|
|
|
+---------------------------------------+---------------------------------------------------------------------------------+
|
|
|
|
| *Python versions and implementations* | .. image:: https://img.shields.io/pypi/pyversions/dependency_injector.svg |
|
|
|
|
| | :target: https://pypi.python.org/pypi/dependency_injector/ |
|
|
|
|
| | :alt: Supported Python versions |
|
|
|
|
| | .. image:: https://img.shields.io/pypi/implementation/dependency_injector.svg |
|
|
|
|
| | :target: https://pypi.python.org/pypi/dependency_injector/ |
|
|
|
|
| | :alt: Supported Python implementations |
|
|
|
|
+---------------------------------------+---------------------------------------------------------------------------------+
|
|
|
|
| *Builds and tests coverage* | .. image:: https://travis-ci.org/ets-labs/dependency_injector.svg?branch=master |
|
|
|
|
| | :target: https://travis-ci.org/ets-labs/dependency_injector |
|
|
|
|
| | :alt: Build Status |
|
|
|
|
| | .. image:: https://coveralls.io/repos/ets-labs/dependency_injector/badge.svg |
|
|
|
|
| | :target: https://coveralls.io/r/ets-labs/dependency_injector |
|
|
|
|
| | :alt: Coverage Status |
|
|
|
|
+---------------------------------------+---------------------------------------------------------------------------------+
|
2015-08-31 16:31:38 +03:00
|
|
|
|
2015-04-03 00:29:00 +03:00
|
|
|
Installation
|
|
|
|
------------
|
|
|
|
|
2015-08-31 16:31:38 +03:00
|
|
|
*Dependency Injector* library is available on PyPi_::
|
2015-04-03 00:29:00 +03:00
|
|
|
|
2015-08-31 16:31:38 +03:00
|
|
|
pip install dependency_injector
|
2015-04-03 00:29:00 +03:00
|
|
|
|
|
|
|
Documentation
|
|
|
|
-------------
|
|
|
|
|
2015-08-31 16:31:38 +03:00
|
|
|
*Dependency Injector* documentation is hosted on ReadTheDocs:
|
2015-04-03 00:29:00 +03:00
|
|
|
|
2015-12-15 18:02:21 +03:00
|
|
|
- `User's guide`_
|
|
|
|
- `API docs`_
|
2015-04-03 00:29:00 +03:00
|
|
|
|
2015-04-03 00:35:22 +03:00
|
|
|
Examples
|
|
|
|
--------
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
2016-02-27 01:13:42 +03:00
|
|
|
"""Pythonic way for Dependency Injection."""
|
|
|
|
|
|
|
|
from dependency_injector import providers
|
|
|
|
from dependency_injector import injections
|
|
|
|
|
|
|
|
|
|
|
|
@providers.DelegatedCallable
|
|
|
|
def get_user_info(user_id):
|
|
|
|
"""Return user info."""
|
|
|
|
raise NotImplementedError()
|
|
|
|
|
|
|
|
|
|
|
|
@providers.Factory
|
|
|
|
@injections.inject(get_user_info=get_user_info)
|
|
|
|
class AuthComponent(object):
|
|
|
|
"""Some authentication component."""
|
|
|
|
|
|
|
|
def __init__(self, get_user_info):
|
|
|
|
"""Initializer."""
|
|
|
|
self.get_user_info = get_user_info
|
|
|
|
|
|
|
|
def authenticate_user(self, token):
|
|
|
|
"""Authenticate user by token."""
|
|
|
|
user_info = self.get_user_info(user_id=token + '1')
|
|
|
|
return user_info
|
|
|
|
|
|
|
|
|
|
|
|
print AuthComponent
|
|
|
|
print get_user_info
|
|
|
|
|
|
|
|
|
|
|
|
@get_user_info.override
|
|
|
|
@providers.DelegatedCallable
|
|
|
|
def get_user_info(user_id):
|
|
|
|
"""Return user info."""
|
|
|
|
return {'user_id': user_id}
|
|
|
|
|
|
|
|
|
|
|
|
print AuthComponent().authenticate_user(token='abc')
|
|
|
|
# {'user_id': 'abc1'}
|
|
|
|
|
|
|
|
One more example with Catalog:
|
|
|
|
|
|
|
|
.. code-block:: python
|
|
|
|
|
|
|
|
"""Pythonic way for Dependency Injection (example with Catalog)."""
|
2015-04-03 00:35:22 +03:00
|
|
|
|
|
|
|
import sqlite3
|
2015-11-23 20:09:35 +03:00
|
|
|
|
|
|
|
from dependency_injector import catalogs
|
|
|
|
from dependency_injector import providers
|
|
|
|
from dependency_injector import injections
|
2015-04-03 00:35:22 +03:00
|
|
|
|
|
|
|
|
2015-10-23 17:04:39 +03:00
|
|
|
class UsersService(object):
|
|
|
|
"""Users service, that has dependency on database."""
|
2015-04-03 00:35:22 +03:00
|
|
|
|
|
|
|
def __init__(self, db):
|
|
|
|
"""Initializer."""
|
|
|
|
self.db = db
|
|
|
|
|
|
|
|
|
2015-10-23 17:04:39 +03:00
|
|
|
class AuthService(object):
|
|
|
|
"""Auth service, that has dependencies on users service and database."""
|
2015-04-03 00:35:22 +03:00
|
|
|
|
2015-10-23 17:04:39 +03:00
|
|
|
def __init__(self, db, users_service):
|
2015-04-03 00:35:22 +03:00
|
|
|
"""Initializer."""
|
|
|
|
self.db = db
|
2015-10-23 17:04:39 +03:00
|
|
|
self.users_service = users_service
|
2015-04-03 00:35:22 +03:00
|
|
|
|
|
|
|
|
2015-11-23 20:09:35 +03:00
|
|
|
class Services(catalogs.DeclarativeCatalog):
|
2015-10-23 17:04:39 +03:00
|
|
|
"""Catalog of service providers."""
|
2015-04-03 00:35:22 +03:00
|
|
|
|
2016-01-11 12:14:37 +03:00
|
|
|
database = providers.Singleton(sqlite3.connect, ':memory:')
|
2015-11-23 20:09:35 +03:00
|
|
|
""":type: providers.Provider -> sqlite3.Connection"""
|
2015-04-03 00:35:22 +03:00
|
|
|
|
2015-11-23 20:09:35 +03:00
|
|
|
users = providers.Factory(UsersService,
|
2016-01-11 12:14:37 +03:00
|
|
|
db=database)
|
2015-11-23 20:09:35 +03:00
|
|
|
""":type: providers.Provider -> UsersService"""
|
2015-04-03 00:35:22 +03:00
|
|
|
|
2015-11-23 20:09:35 +03:00
|
|
|
auth = providers.Factory(AuthService,
|
2016-01-11 12:14:37 +03:00
|
|
|
db=database,
|
|
|
|
users_service=users)
|
2015-11-23 20:09:35 +03:00
|
|
|
""":type: providers.Provider -> AuthService"""
|
2015-04-03 00:35:22 +03:00
|
|
|
|
|
|
|
|
2015-10-23 17:04:39 +03:00
|
|
|
# Retrieving catalog providers:
|
|
|
|
users_service = Services.users()
|
|
|
|
auth_service = Services.auth()
|
2015-04-03 00:35:22 +03:00
|
|
|
|
2015-10-23 17:04:39 +03:00
|
|
|
# Making some asserts:
|
|
|
|
assert users_service.db is auth_service.db is Services.database()
|
|
|
|
assert isinstance(auth_service.users_service, UsersService)
|
|
|
|
assert users_service is not Services.users()
|
|
|
|
assert auth_service is not Services.auth()
|
2015-04-03 00:35:22 +03:00
|
|
|
|
|
|
|
|
2015-10-23 17:04:39 +03:00
|
|
|
# Making some "inline" injections:
|
2015-11-23 20:09:35 +03:00
|
|
|
@injections.inject(users_service=Services.users)
|
|
|
|
@injections.inject(auth_service=Services.auth)
|
|
|
|
@injections.inject(database=Services.database)
|
2015-10-23 17:04:39 +03:00
|
|
|
def example(users_service, auth_service, database):
|
2015-08-05 17:33:12 +03:00
|
|
|
"""Example callback."""
|
2015-10-23 17:04:39 +03:00
|
|
|
assert users_service.db is auth_service.db
|
|
|
|
assert auth_service.db is database
|
|
|
|
assert database is Services.database()
|
2015-04-03 00:35:22 +03:00
|
|
|
|
|
|
|
|
2015-10-23 17:04:39 +03:00
|
|
|
# Making a call of decorated callback:
|
2015-04-03 00:35:22 +03:00
|
|
|
example()
|
|
|
|
|
2016-01-11 12:14:37 +03:00
|
|
|
|
|
|
|
# Overriding auth service provider and making some asserts:
|
|
|
|
class ExtendedAuthService(AuthService):
|
|
|
|
"""Extended version of auth service."""
|
|
|
|
|
|
|
|
def __init__(self, db, users_service, ttl):
|
|
|
|
"""Initializer."""
|
|
|
|
self.ttl = ttl
|
|
|
|
super(ExtendedAuthService, self).__init__(db=db,
|
|
|
|
users_service=users_service)
|
|
|
|
|
|
|
|
|
|
|
|
Services.auth.override(providers.Factory(ExtendedAuthService,
|
|
|
|
db=Services.database,
|
|
|
|
users_service=Services.users,
|
|
|
|
ttl=3600))
|
|
|
|
|
|
|
|
|
|
|
|
auth_service = Services.auth()
|
|
|
|
|
|
|
|
assert isinstance(auth_service, ExtendedAuthService)
|
|
|
|
|
2015-08-31 16:31:38 +03:00
|
|
|
You can get more *Dependency Injector* examples in ``/examples`` directory on
|
2015-04-03 00:40:03 +03:00
|
|
|
GitHub:
|
|
|
|
|
2016-01-11 11:53:35 +03:00
|
|
|
https://github.com/ets-labs/dependency_injector
|
2015-04-03 00:35:22 +03:00
|
|
|
|
|
|
|
|
2015-04-03 00:33:28 +03:00
|
|
|
Feedback
|
2015-04-03 00:35:22 +03:00
|
|
|
--------
|
2015-04-03 00:33:28 +03:00
|
|
|
|
|
|
|
Feel free to post questions, bugs, feature requests, proposals etc. on
|
2015-08-31 16:31:38 +03:00
|
|
|
*Dependency Injector* GitHub Issues:
|
2015-04-03 00:33:28 +03:00
|
|
|
|
2016-01-11 11:53:35 +03:00
|
|
|
https://github.com/ets-labs/dependency_injector/issues
|
2015-04-03 00:33:28 +03:00
|
|
|
|
|
|
|
Your feedback is quite important!
|
|
|
|
|
2015-04-03 00:29:00 +03:00
|
|
|
|
2015-08-31 16:31:38 +03:00
|
|
|
.. _PyPi: https://pypi.python.org/pypi/dependency_injector
|
2015-12-15 18:02:21 +03:00
|
|
|
.. _User's guide: http://dependency_injector.readthedocs.org/en/stable/
|
|
|
|
.. _API docs: http://dependency-injector.readthedocs.org/en/stable/api/
|
2015-05-12 16:18:37 +03:00
|
|
|
.. _SLOC: http://en.wikipedia.org/wiki/Source_lines_of_code
|
|
|
|
.. _SOLID: http://en.wikipedia.org/wiki/SOLID_%28object-oriented_design%29
|
|
|
|
.. _IoC: http://en.wikipedia.org/wiki/Inversion_of_control
|
|
|
|
.. _dependency injection: http://en.wikipedia.org/wiki/Dependency_injection
|