modern-di 0.3.0__py3-none-any.whl → 0.4.0__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of modern-di might be problematic. Click here for more details.

modern_di/graph.py CHANGED
@@ -1,7 +1,7 @@
1
1
  import typing
2
2
 
3
3
  from modern_di import Container
4
- from modern_di.providers import AbstractProvider, BaseCreatorProvider
4
+ from modern_di.providers.abstract import AbstractCreatorProvider, AbstractProvider
5
5
 
6
6
 
7
7
  if typing.TYPE_CHECKING:
@@ -29,11 +29,11 @@ class BaseGraph:
29
29
  @classmethod
30
30
  async def async_resolve_creators(cls, container: Container) -> None:
31
31
  for provider in cls.get_providers().values():
32
- if isinstance(provider, BaseCreatorProvider) and provider.scope == container.scope:
32
+ if isinstance(provider, AbstractCreatorProvider) and provider.scope == container.scope:
33
33
  await provider.async_resolve(container)
34
34
 
35
35
  @classmethod
36
36
  def sync_resolve_creators(cls, container: Container) -> None:
37
37
  for provider in cls.get_providers().values():
38
- if isinstance(provider, BaseCreatorProvider) and provider.scope == container.scope:
38
+ if isinstance(provider, AbstractCreatorProvider) and provider.scope == container.scope:
39
39
  provider.sync_resolve(container)
@@ -1,15 +1,16 @@
1
- from modern_di.providers.base import AbstractProvider, BaseCreatorProvider
2
- from modern_di.providers.context_adapter import ContextAdapter
1
+ from modern_di.providers.dict import Dict
3
2
  from modern_di.providers.factory import Factory
3
+ from modern_di.providers.list import List
4
4
  from modern_di.providers.resource import Resource
5
+ from modern_di.providers.selector import Selector
5
6
  from modern_di.providers.singleton import Singleton
6
7
 
7
8
 
8
9
  __all__ = [
9
- "AbstractProvider",
10
- "BaseCreatorProvider",
11
- "ContextAdapter",
12
10
  "Factory",
11
+ "Dict",
12
+ "List",
13
+ "Selector",
13
14
  "Singleton",
14
15
  "Resource",
15
16
  ]
@@ -27,18 +27,25 @@ class AbstractProvider(typing.Generic[T_co], abc.ABC):
27
27
  def sync_resolve(self, container: Container) -> T_co:
28
28
  """Resolve dependency synchronously."""
29
29
 
30
+ @property
31
+ def cast(self) -> T_co:
32
+ return typing.cast(T_co, self)
33
+
34
+ def _check_providers_scope(self, providers: typing.Iterable[typing.Any]) -> None:
35
+ if any(x.scope > self.scope for x in providers if isinstance(x, AbstractProvider)):
36
+ msg = "Scope of dependency cannot be more than scope of dependent"
37
+ raise RuntimeError(msg)
38
+
39
+
40
+ class AbstractOverrideProvider(AbstractProvider[T_co], abc.ABC):
30
41
  def override(self, override_object: object, container: Container) -> None:
31
42
  container.override(self.provider_id, override_object)
32
43
 
33
44
  def reset_override(self, container: Container) -> None:
34
45
  container.reset_override(self.provider_id)
35
46
 
36
- @property
37
- def cast(self) -> T_co:
38
- return typing.cast(T_co, self)
39
-
40
47
 
41
- class BaseCreatorProvider(AbstractProvider[T_co], abc.ABC):
48
+ class AbstractCreatorProvider(AbstractOverrideProvider[T_co], abc.ABC):
42
49
  BASE_SLOTS: typing.ClassVar = [*AbstractProvider.BASE_SLOTS, "_args", "_kwargs", "_creator"]
43
50
 
44
51
  def __init__(
@@ -49,11 +56,7 @@ class BaseCreatorProvider(AbstractProvider[T_co], abc.ABC):
49
56
  **kwargs: P.kwargs,
50
57
  ) -> None:
51
58
  super().__init__(scope)
52
-
53
- if any(x.scope > self.scope for x in itertools.chain(args, kwargs.values()) if isinstance(x, AbstractProvider)):
54
- msg = "Scope of dependency cannot be more than scope of dependent"
55
- raise RuntimeError(msg)
56
-
59
+ self._check_providers_scope(itertools.chain(args, kwargs.values()))
57
60
  self._creator: typing.Final = creator
58
61
  self._args: typing.Final = args
59
62
  self._kwargs: typing.Final = kwargs
@@ -0,0 +1,23 @@
1
+ import enum
2
+ import typing
3
+
4
+ from modern_di import Container
5
+ from modern_di.providers.abstract import AbstractProvider
6
+
7
+
8
+ T_co = typing.TypeVar("T_co", covariant=True)
9
+
10
+
11
+ class Dict(AbstractProvider[dict[str, T_co]]):
12
+ __slots__ = [*AbstractProvider.BASE_SLOTS, "_providers"]
13
+
14
+ def __init__(self, scope: enum.IntEnum, **providers: AbstractProvider[T_co]) -> None:
15
+ super().__init__(scope)
16
+ self._check_providers_scope(providers.values())
17
+ self._providers: typing.Final = providers
18
+
19
+ async def async_resolve(self, container: Container) -> dict[str, T_co]:
20
+ return {key: await provider.async_resolve(container) for key, provider in self._providers.items()}
21
+
22
+ def sync_resolve(self, container: Container) -> dict[str, T_co]:
23
+ return {key: provider.sync_resolve(container) for key, provider in self._providers.items()}
@@ -2,15 +2,15 @@ import enum
2
2
  import typing
3
3
 
4
4
  from modern_di import Container
5
- from modern_di.providers import BaseCreatorProvider
5
+ from modern_di.providers.abstract import AbstractCreatorProvider
6
6
 
7
7
 
8
8
  T_co = typing.TypeVar("T_co", covariant=True)
9
9
  P = typing.ParamSpec("P")
10
10
 
11
11
 
12
- class Factory(BaseCreatorProvider[T_co]):
13
- __slots__ = [*BaseCreatorProvider.BASE_SLOTS, "_creator"]
12
+ class Factory(AbstractCreatorProvider[T_co]):
13
+ __slots__ = [*AbstractCreatorProvider.BASE_SLOTS, "_creator"]
14
14
 
15
15
  def __init__(
16
16
  self,
@@ -0,0 +1,23 @@
1
+ import enum
2
+ import typing
3
+
4
+ from modern_di import Container
5
+ from modern_di.providers.abstract import AbstractProvider
6
+
7
+
8
+ T_co = typing.TypeVar("T_co", covariant=True)
9
+
10
+
11
+ class List(AbstractProvider[list[T_co]]):
12
+ __slots__ = [*AbstractProvider.BASE_SLOTS, "_providers"]
13
+
14
+ def __init__(self, scope: enum.IntEnum, *providers: AbstractProvider[T_co]) -> None:
15
+ super().__init__(scope)
16
+ self._check_providers_scope(providers)
17
+ self._providers: typing.Final = providers
18
+
19
+ async def async_resolve(self, container: Container) -> list[T_co]:
20
+ return [await x.async_resolve(container) for x in self._providers]
21
+
22
+ def sync_resolve(self, container: Container) -> list[T_co]:
23
+ return [x.sync_resolve(container) for x in self._providers]
@@ -4,15 +4,15 @@ import inspect
4
4
  import typing
5
5
 
6
6
  from modern_di import Container
7
- from modern_di.providers import BaseCreatorProvider
7
+ from modern_di.providers.abstract import AbstractCreatorProvider
8
8
 
9
9
 
10
10
  T_co = typing.TypeVar("T_co", covariant=True)
11
11
  P = typing.ParamSpec("P")
12
12
 
13
13
 
14
- class Resource(BaseCreatorProvider[T_co]):
15
- __slots__ = [*BaseCreatorProvider.BASE_SLOTS, "_creator", "_args", "_kwargs", "_is_async"]
14
+ class Resource(AbstractCreatorProvider[T_co]):
15
+ __slots__ = [*AbstractCreatorProvider.BASE_SLOTS, "_creator", "_args", "_kwargs", "_is_async"]
16
16
 
17
17
  def _is_creator_async(
18
18
  self,
@@ -0,0 +1,39 @@
1
+ import enum
2
+ import typing
3
+
4
+ from modern_di import Container
5
+ from modern_di.providers.abstract import AbstractProvider
6
+
7
+
8
+ T_co = typing.TypeVar("T_co", covariant=True)
9
+ P = typing.ParamSpec("P")
10
+
11
+
12
+ class Selector(AbstractProvider[T_co]):
13
+ __slots__ = [*AbstractProvider.BASE_SLOTS, "_function", "_providers"]
14
+
15
+ def __init__(
16
+ self, scope: enum.IntEnum, function: typing.Callable[..., str], **providers: AbstractProvider[T_co]
17
+ ) -> None:
18
+ super().__init__(scope)
19
+ self._check_providers_scope(providers.values())
20
+ self._function: typing.Final = function
21
+ self._providers: typing.Final = providers
22
+
23
+ async def async_resolve(self, container: Container) -> T_co:
24
+ container = container.find_container(self.scope)
25
+ selected_key = self._function(**container.context)
26
+ if selected_key not in self._providers:
27
+ msg = f"No provider matches {selected_key}"
28
+ raise RuntimeError(msg)
29
+
30
+ return await self._providers[selected_key].async_resolve(container)
31
+
32
+ def sync_resolve(self, container: Container) -> T_co:
33
+ container = container.find_container(self.scope)
34
+ selected_key = self._function(**container.context)
35
+ if selected_key not in self._providers:
36
+ msg = f"No provider matches {selected_key}"
37
+ raise RuntimeError(msg)
38
+
39
+ return self._providers[selected_key].sync_resolve(container)
@@ -2,15 +2,15 @@ import enum
2
2
  import typing
3
3
 
4
4
  from modern_di import Container
5
- from modern_di.providers import BaseCreatorProvider
5
+ from modern_di.providers.abstract import AbstractCreatorProvider
6
6
 
7
7
 
8
8
  T_co = typing.TypeVar("T_co", covariant=True)
9
9
  P = typing.ParamSpec("P")
10
10
 
11
11
 
12
- class Singleton(BaseCreatorProvider[T_co]):
13
- __slots__ = [*BaseCreatorProvider.BASE_SLOTS, "_creator"]
12
+ class Singleton(AbstractCreatorProvider[T_co]):
13
+ __slots__ = [*AbstractCreatorProvider.BASE_SLOTS, "_creator"]
14
14
 
15
15
  def __init__(
16
16
  self,
@@ -0,0 +1,39 @@
1
+ Metadata-Version: 2.3
2
+ Name: modern-di
3
+ Version: 0.4.0
4
+ Summary: Dependency Injection framework with IOC-container and scopes
5
+ Project-URL: repository, https://github.com/modern-python/modern-di
6
+ Project-URL: docs, https://modern-di.readthedocs.io
7
+ Author-email: Artur Shiriev <me@shiriev.ru>
8
+ License-Expression: MIT
9
+ License-File: LICENSE
10
+ Keywords: dependency injector,di,ioc-container,mocks,python
11
+ Classifier: Programming Language :: Python :: 3.10
12
+ Classifier: Programming Language :: Python :: 3.11
13
+ Classifier: Programming Language :: Python :: 3.12
14
+ Classifier: Programming Language :: Python :: 3.13
15
+ Classifier: Topic :: Software Development :: Libraries
16
+ Classifier: Typing :: Typed
17
+ Requires-Python: <4,>=3.10
18
+ Description-Content-Type: text/markdown
19
+
20
+ "Modern-DI"
21
+ ==
22
+
23
+ | Project | Badges |
24
+ |-------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
25
+ | common | [![MyPy Strict](https://img.shields.io/badge/mypy-strict-blue)](https://mypy.readthedocs.io/en/stable/getting_started.html#strict-mode-and-configuration) [![GitHub stars](https://img.shields.io/github/stars/modern-python/modern-di)](https://github.com/modern-python/modern-di/stargazers) |
26
+ | modern-di | [![Supported versions](https://img.shields.io/pypi/pyversions/modern-di.svg)](https://pypi.python.org/pypi/modern-di ) [![downloads](https://img.shields.io/pypi/dm/modern-di.svg)](https://pypistats.org/packages/modern-di) |
27
+ | modern-di-fastapi | [![Supported versions](https://img.shields.io/pypi/pyversions/modern-di-fastapi.svg)](https://pypi.python.org/pypi/modern-di-fastapi) [![downloads](https://img.shields.io/pypi/dm/modern-di-fastapi.svg)](https://pypistats.org/packages/modern-di-fastapi) |
28
+
29
+ Dependency injection framework for Python inspired by `dependency-injector` and `dishka`.
30
+
31
+ It is in early development state and gives you the following:
32
+ - DI framework with IOC-container and scopes.
33
+ - Async and sync resolving.
34
+ - Python 3.10-3.13 support.
35
+ - Full coverage by types annotations (mypy in strict mode).
36
+ - Overriding dependencies for tests.
37
+ - Package with zero dependencies.
38
+
39
+ 📚 [Documentation](https://modern-di.readthedocs.io)
@@ -0,0 +1,18 @@
1
+ modern_di/__init__.py,sha256=L01VkzSJiV0d0FPrh1DZ-Wy5mUmoG6X-oLz7xYxtehI,194
2
+ modern_di/container.py,sha256=xAUiQ10nI_a0NcvedHJ-IFCBzr3HG7TtuJ6mwPiVhmA,3998
3
+ modern_di/graph.py,sha256=X60wtG3Mqus_5YZNiZlQuXoHODBp7rYl_IHJs7GzSQM,1356
4
+ modern_di/provider_state.py,sha256=5Bl_iYEpXjMqoWZJ4op2-axo4Z8nR_vYbLVHL_u5R0c,1206
5
+ modern_di/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
6
+ modern_di/scope.py,sha256=nnrTLnFT_dMDLVIwvJHcGlJsB2k1WIHusrRSGQIMEsg,97
7
+ modern_di/providers/__init__.py,sha256=qiSLdMtoN_Bt33UP0tuSERqzDiW7PdHaVBnQ8h11M_0,388
8
+ modern_di/providers/abstract.py,sha256=UMj4CRn-JfGZfiveWFTkH7V92h4UXS4eYf8noZqPWGQ,3107
9
+ modern_di/providers/dict.py,sha256=nCU9iaqteYHDbILAfhrdnbMgS9_emE4MS7Xn2VoUlPo,858
10
+ modern_di/providers/factory.py,sha256=-QXh_qoJvYIY5kxKPkGYup5pqQvS9HSietI0MqKOb_Q,1185
11
+ modern_di/providers/list.py,sha256=3hx34RfBRmqzh-cT5D6wSTDJPkBGMK_ul4n9gQz-o9M,769
12
+ modern_di/providers/resource.py,sha256=nKy8_Wfn2cAIvlu_qGjGanmlgrkHQUsZxBNNR6p7OuE,3649
13
+ modern_di/providers/selector.py,sha256=RQbHD2-Liw-TGqu6UELbfCzXYuqxiO_Mg1tLyF3mKQo,1419
14
+ modern_di/providers/singleton.py,sha256=7XBNhVzhV5Rh_F7iWZx8is7i7_PuctQ9thKeqIkjnTs,1999
15
+ modern_di-0.4.0.dist-info/METADATA,sha256=AU2E9RJaaUQppUM786712IJKNobdLmsn05DHccQvM5s,2767
16
+ modern_di-0.4.0.dist-info/WHEEL,sha256=1yFddiXMmvYK7QYTqtRNtX66WJ0Mz8PYEiEUoOUUxRY,87
17
+ modern_di-0.4.0.dist-info/licenses/LICENSE,sha256=aA5_eJwDKqnGvL7PbkPb9x5f-VGIqZ9cvWWkeGqeD90,1070
18
+ modern_di-0.4.0.dist-info/RECORD,,
@@ -1,31 +0,0 @@
1
- import enum
2
- import typing
3
-
4
- from modern_di import Container
5
- from modern_di.providers import AbstractProvider
6
-
7
-
8
- T_co = typing.TypeVar("T_co", covariant=True)
9
- P = typing.ParamSpec("P")
10
-
11
-
12
- class ContextAdapter(AbstractProvider[T_co]):
13
- __slots__ = [*AbstractProvider.BASE_SLOTS, "_function"]
14
-
15
- def __init__(
16
- self,
17
- scope: enum.IntEnum,
18
- function: typing.Callable[..., T_co],
19
- ) -> None:
20
- super().__init__(scope)
21
- self._function = function
22
-
23
- async def async_resolve(self, container: Container) -> T_co:
24
- return self.sync_resolve(container)
25
-
26
- def sync_resolve(self, container: Container) -> T_co:
27
- container = container.find_container(self.scope)
28
- if (override := container.fetch_override(self.provider_id)) is not None:
29
- return typing.cast(T_co, override)
30
-
31
- return self._function(**container.context)
@@ -1,37 +0,0 @@
1
- Metadata-Version: 2.3
2
- Name: modern-di
3
- Version: 0.3.0
4
- Summary: Dependency Injection framework with IOC-container and scopes
5
- Project-URL: repository, https://github.com/modern-python/modern-di
6
- Project-URL: docs, https://modern-di.readthedocs.io
7
- Author-email: Artur Shiriev <me@shiriev.ru>
8
- License-Expression: MIT
9
- License-File: LICENSE
10
- Keywords: dependency injector,di,ioc-container,mocks,python
11
- Classifier: Programming Language :: Python :: 3.10
12
- Classifier: Programming Language :: Python :: 3.11
13
- Classifier: Programming Language :: Python :: 3.12
14
- Classifier: Programming Language :: Python :: 3.13
15
- Classifier: Topic :: Software Development :: Libraries
16
- Classifier: Typing :: Typed
17
- Requires-Python: <4,>=3.10
18
- Description-Content-Type: text/markdown
19
-
20
- "Modern-DI"
21
- ==
22
- [![MyPy Strict](https://img.shields.io/badge/mypy-strict-blue)](https://mypy.readthedocs.io/en/stable/getting_started.html#strict-mode-and-configuration)
23
- [![Supported versions](https://img.shields.io/pypi/pyversions/modern-di.svg)](https://pypi.python.org/pypi/modern-di)
24
- [![downloads](https://img.shields.io/pypi/dm/modern-di.svg)](https://pypistats.org/packages/modern-di)
25
- [![GitHub stars](https://img.shields.io/github/stars/modern-python/modern-di)](https://github.com/modern-python/modern-di/stargazers)
26
-
27
- Dependency injection framework for Python inspired by `dependency-injector` and `dishka`.
28
-
29
- It is in early development state and gives you the following:
30
- - DI framework with IOC-container and scopes.
31
- - Async and sync resolving.
32
- - Python 3.10-3.13 support.
33
- - Full coverage by types annotations (mypy in strict mode).
34
- - Overriding dependencies for tests.
35
- - Package with zero dependencies.
36
-
37
- 📚 [Documentation](https://modern-di.readthedocs.io)
@@ -1,16 +0,0 @@
1
- modern_di/__init__.py,sha256=L01VkzSJiV0d0FPrh1DZ-Wy5mUmoG6X-oLz7xYxtehI,194
2
- modern_di/container.py,sha256=xAUiQ10nI_a0NcvedHJ-IFCBzr3HG7TtuJ6mwPiVhmA,3998
3
- modern_di/graph.py,sha256=TaFlc7So3aGRjmv6XXVz4CawNCIadoWuTun4OPRC27M,1335
4
- modern_di/provider_state.py,sha256=5Bl_iYEpXjMqoWZJ4op2-axo4Z8nR_vYbLVHL_u5R0c,1206
5
- modern_di/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
6
- modern_di/scope.py,sha256=nnrTLnFT_dMDLVIwvJHcGlJsB2k1WIHusrRSGQIMEsg,97
7
- modern_di/providers/__init__.py,sha256=n389r5MlYTaVQ5nFYsy_keZqLasnW6uOKRYj0_F-aoY,425
8
- modern_di/providers/base.py,sha256=-cXfJdb8_XpiZjxYw_c9vDD3YY6FSH1jAokynApYeX0,2897
9
- modern_di/providers/context_adapter.py,sha256=61SmY7jAx0d0akdREmDhYjeFJUeiBB0zghxPyd2-Ads,885
10
- modern_di/providers/factory.py,sha256=tx2Inik1IqQcDY8gsbQ1jWZf7hCuJbw_XC9h-RKxRVw,1164
11
- modern_di/providers/resource.py,sha256=my4hDJ1i-PZTCDWTC1WT7Cd66ki-d2b61nVdfLG-vPQ,3628
12
- modern_di/providers/singleton.py,sha256=30K2JOXSV3h_aPckXTIC9g5iGNmOfZpoBOeEX9RtYu4,1978
13
- modern_di-0.3.0.dist-info/METADATA,sha256=Dc-vTMRaGuk3GFOBjbjD9XPkxM7bpK2HDKynKBBUEJo,1715
14
- modern_di-0.3.0.dist-info/WHEEL,sha256=1yFddiXMmvYK7QYTqtRNtX66WJ0Mz8PYEiEUoOUUxRY,87
15
- modern_di-0.3.0.dist-info/licenses/LICENSE,sha256=aA5_eJwDKqnGvL7PbkPb9x5f-VGIqZ9cvWWkeGqeD90,1070
16
- modern_di-0.3.0.dist-info/RECORD,,