modern-di 0.9.0__tar.gz → 0.11.0__tar.gz

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.

Files changed (22) hide show
  1. {modern_di-0.9.0 → modern_di-0.11.0}/PKG-INFO +24 -6
  2. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/container.py +1 -1
  3. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/__init__.py +2 -0
  4. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/abstract.py +19 -21
  5. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/factory.py +7 -3
  6. modern_di-0.11.0/modern_di/providers/injected_factory.py +45 -0
  7. modern_di-0.11.0/modern_di/providers/object.py +27 -0
  8. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/resource.py +1 -1
  9. modern_di-0.9.0/modern_di/providers/injected_factory.py +0 -29
  10. {modern_di-0.9.0 → modern_di-0.11.0}/.gitignore +0 -0
  11. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/__init__.py +0 -0
  12. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/graph.py +0 -0
  13. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/provider_state.py +0 -0
  14. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/container_provider.py +0 -0
  15. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/context_adapter.py +0 -0
  16. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/dict.py +0 -0
  17. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/list.py +0 -0
  18. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/selector.py +0 -0
  19. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/providers/singleton.py +0 -0
  20. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/py.typed +0 -0
  21. {modern_di-0.9.0 → modern_di-0.11.0}/modern_di/scope.py +0 -0
  22. {modern_di-0.9.0 → modern_di-0.11.0}/pyproject.toml +0 -0
@@ -1,11 +1,11 @@
1
- Metadata-Version: 2.3
1
+ Metadata-Version: 2.4
2
2
  Name: modern-di
3
- Version: 0.9.0
3
+ Version: 0.11.0
4
4
  Summary: Dependency Injection framework with IOC-container and scopes
5
5
  Project-URL: repository, https://github.com/modern-python/modern-di
6
6
  Project-URL: docs, https://modern-di.readthedocs.io
7
7
  Author-email: Artur Shiriev <me@shiriev.ru>
8
- License: MIT
8
+ License-Expression: MIT
9
9
  Keywords: DI,dependency injector,ioc-container,mocks,python
10
10
  Classifier: Programming Language :: Python :: 3.10
11
11
  Classifier: Programming Language :: Python :: 3.11
@@ -40,7 +40,17 @@ It is in development state yet and gives you the following:
40
40
 
41
41
  📚 [Documentation](https://modern-di.readthedocs.io)
42
42
 
43
- ## Describe resources and classes:
43
+ # Quickstart
44
+
45
+ ## 1. Install `modern-di` using your favorite tool:
46
+
47
+ ```shell
48
+ pip install modern-di
49
+ uv add modern-di
50
+ poetry add modern-di
51
+ ```
52
+
53
+ ## 2. Describe resources and classes:
44
54
  ```python
45
55
  import dataclasses
46
56
  import logging
@@ -74,7 +84,7 @@ class DependentFactory:
74
84
  async_resource: str
75
85
  ```
76
86
 
77
- ## Describe dependencies graph (IoC-container)
87
+ ## 3. Describe dependencies graph
78
88
  ```python
79
89
  from modern_di import BaseGraph, Scope, providers
80
90
 
@@ -91,7 +101,15 @@ class Dependencies(BaseGraph):
91
101
  )
92
102
  ```
93
103
 
94
- ## Create container and resolve dependencies in your code
104
+ ## 4.1. Integrate with your framework
105
+
106
+ For now there are integration for the following frameworks:
107
+ 1. [FastAPI](https://modern-di.readthedocs.io/en/latest/integrations/fastapi.html)
108
+ 2. [LiteStar](https://modern-di.readthedocs.io/en/latest/integrations/litestar.html)
109
+
110
+ ## 4.2. Or use `modern-di` without integrations
111
+
112
+ Create container and resolve dependencies in your code
95
113
  ```python
96
114
  from modern_di import Container, Scope
97
115
 
@@ -13,7 +13,7 @@ if typing.TYPE_CHECKING:
13
13
  T_co = typing.TypeVar("T_co", covariant=True)
14
14
 
15
15
 
16
- class Container(contextlib.AbstractAsyncContextManager["Container"]):
16
+ class Container(contextlib.AbstractAsyncContextManager["Container"], contextlib.AbstractContextManager["Container"]):
17
17
  __slots__ = (
18
18
  "_is_async",
19
19
  "_overrides",
@@ -4,6 +4,7 @@ from modern_di.providers.context_adapter import ContextAdapter
4
4
  from modern_di.providers.dict import Dict
5
5
  from modern_di.providers.factory import Factory
6
6
  from modern_di.providers.list import List
7
+ from modern_di.providers.object import Object
7
8
  from modern_di.providers.resource import Resource
8
9
  from modern_di.providers.selector import Selector
9
10
  from modern_di.providers.singleton import Singleton
@@ -16,6 +17,7 @@ __all__ = [
16
17
  "Dict",
17
18
  "Factory",
18
19
  "List",
20
+ "Object",
19
21
  "Resource",
20
22
  "Selector",
21
23
  "Singleton",
@@ -61,31 +61,29 @@ class AbstractCreatorProvider(AbstractOverrideProvider[T_co], abc.ABC):
61
61
  self._args: typing.Final = args
62
62
  self._kwargs: typing.Final = kwargs
63
63
 
64
+ def _sync_resolve_args(self, container: Container) -> list[typing.Any]:
65
+ return [x.sync_resolve(container) if isinstance(x, AbstractProvider) else x for x in self._args]
66
+
67
+ def _sync_resolve_kwargs(self, container: Container) -> dict[str, typing.Any]:
68
+ return {k: v.sync_resolve(container) if isinstance(v, AbstractProvider) else v for k, v in self._kwargs.items()}
69
+
64
70
  def _sync_build_creator(self, container: Container) -> typing.Any: # noqa: ANN401
65
71
  return self._creator(
66
- *typing.cast(
67
- P.args, [x.sync_resolve(container) if isinstance(x, AbstractProvider) else x for x in self._args]
68
- ),
69
- **typing.cast(
70
- P.kwargs,
71
- {
72
- k: v.sync_resolve(container) if isinstance(v, AbstractProvider) else v
73
- for k, v in self._kwargs.items()
74
- },
75
- ),
72
+ *typing.cast(P.args, self._sync_resolve_args(container)),
73
+ **typing.cast(P.kwargs, self._sync_resolve_kwargs(container)),
76
74
  )
77
75
 
76
+ async def _async_resolve_args(self, container: Container) -> list[typing.Any]:
77
+ return [await x.async_resolve(container) if isinstance(x, AbstractProvider) else x for x in self._args]
78
+
79
+ async def _async_resolve_kwargs(self, container: Container) -> dict[str, typing.Any]:
80
+ return {
81
+ k: await v.async_resolve(container) if isinstance(v, AbstractProvider) else v
82
+ for k, v in self._kwargs.items()
83
+ }
84
+
78
85
  async def _async_build_creator(self, container: Container) -> typing.Any: # noqa: ANN401
79
86
  return self._creator(
80
- *typing.cast(
81
- P.args,
82
- [await x.async_resolve(container) if isinstance(x, AbstractProvider) else x for x in self._args],
83
- ),
84
- **typing.cast(
85
- P.kwargs,
86
- {
87
- k: await v.async_resolve(container) if isinstance(v, AbstractProvider) else v
88
- for k, v in self._kwargs.items()
89
- },
90
- ),
87
+ *typing.cast(P.args, await self._async_resolve_args(container)),
88
+ **typing.cast(P.kwargs, await self._async_resolve_kwargs(container)),
91
89
  )
@@ -3,7 +3,7 @@ import typing
3
3
 
4
4
  from modern_di import Container
5
5
  from modern_di.providers.abstract import AbstractCreatorProvider
6
- from modern_di.providers.injected_factory import InjectedFactory
6
+ from modern_di.providers.injected_factory import AsyncInjectedFactory, SyncInjectedFactory
7
7
 
8
8
 
9
9
  T_co = typing.TypeVar("T_co", covariant=True)
@@ -23,8 +23,12 @@ class Factory(AbstractCreatorProvider[T_co]):
23
23
  super().__init__(scope, creator, *args, **kwargs)
24
24
 
25
25
  @property
26
- def factory_provider(self) -> InjectedFactory[T_co]:
27
- return InjectedFactory(self)
26
+ def sync_provider(self) -> SyncInjectedFactory[T_co]:
27
+ return SyncInjectedFactory(self)
28
+
29
+ @property
30
+ def async_provider(self) -> AsyncInjectedFactory[T_co]:
31
+ return AsyncInjectedFactory(self)
28
32
 
29
33
  async def async_resolve(self, container: Container) -> T_co:
30
34
  container = container.find_container(self.scope)
@@ -0,0 +1,45 @@
1
+ import functools
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 SyncInjectedFactory(AbstractProvider[T_co]):
13
+ __slots__ = [*AbstractProvider.BASE_SLOTS, "_factory_provider"]
14
+
15
+ def __init__(self, factory_provider: AbstractProvider[T_co]) -> None:
16
+ super().__init__(factory_provider.scope)
17
+ self._factory_provider = factory_provider
18
+
19
+ async def async_resolve(self, container: Container) -> typing.Callable[[], T_co]: # type: ignore[override]
20
+ return self.sync_resolve(container)
21
+
22
+ def sync_resolve(self, container: Container) -> typing.Callable[[], T_co]: # type: ignore[override]
23
+ return functools.partial(self._factory_provider.sync_resolve, container)
24
+
25
+ @property
26
+ def cast(self) -> typing.Callable[[], T_co]: # type: ignore[override]
27
+ return typing.cast(typing.Callable[[], T_co], self)
28
+
29
+
30
+ class AsyncInjectedFactory(AbstractProvider[T_co]):
31
+ __slots__ = [*AbstractProvider.BASE_SLOTS, "_factory_provider"]
32
+
33
+ def __init__(self, factory_provider: AbstractProvider[T_co]) -> None:
34
+ super().__init__(factory_provider.scope)
35
+ self._factory_provider = factory_provider
36
+
37
+ async def async_resolve(self, container: Container) -> typing.Callable[[], typing.Awaitable[T_co]]: # type: ignore[override]
38
+ return self.sync_resolve(container)
39
+
40
+ def sync_resolve(self, container: Container) -> typing.Callable[[], typing.Awaitable[T_co]]: # type: ignore[override]
41
+ return functools.partial(self._factory_provider.async_resolve, container)
42
+
43
+ @property
44
+ def cast(self) -> typing.Callable[[], typing.Awaitable[T_co]]: # type: ignore[override]
45
+ return typing.cast(typing.Callable[[], typing.Awaitable[T_co]], self)
@@ -0,0 +1,27 @@
1
+ import enum
2
+ import typing
3
+
4
+ from modern_di import Container
5
+ from modern_di.providers.abstract import AbstractOverrideProvider
6
+
7
+
8
+ T_co = typing.TypeVar("T_co", covariant=True)
9
+ P = typing.ParamSpec("P")
10
+
11
+
12
+ class Object(AbstractOverrideProvider[T_co]):
13
+ __slots__ = [*AbstractOverrideProvider.BASE_SLOTS, "_obj"]
14
+
15
+ def __init__(self, scope: enum.IntEnum, obj: T_co) -> None:
16
+ super().__init__(scope)
17
+ self._obj: typing.Final = obj
18
+
19
+ async def async_resolve(self, container: Container) -> T_co:
20
+ return self.sync_resolve(container)
21
+
22
+ def sync_resolve(self, container: Container) -> T_co:
23
+ container = container.find_container(self.scope)
24
+ if (override := container.fetch_override(self.provider_id)) is not None:
25
+ return typing.cast(T_co, override)
26
+
27
+ return self._obj
@@ -1,4 +1,4 @@
1
- import contextlib
1
+ import contextlib # noqa: A005
2
2
  import enum
3
3
  import inspect
4
4
  import typing
@@ -1,29 +0,0 @@
1
- import functools
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 InjectedFactory(AbstractProvider[T_co]):
13
- __slots__ = [*AbstractProvider.BASE_SLOTS, "_factory_provider"]
14
-
15
- def __init__(self, factory_provider: AbstractProvider[T_co]) -> None:
16
- super().__init__(factory_provider.scope)
17
- self._factory_provider = factory_provider
18
-
19
- async def async_resolve(self, container: Container) -> typing.Callable[[], T_co]: # type: ignore[override]
20
- await self._factory_provider.async_resolve(container)
21
- return functools.partial(self._factory_provider.sync_resolve, container)
22
-
23
- def sync_resolve(self, container: Container) -> typing.Callable[[], T_co]: # type: ignore[override]
24
- self._factory_provider.sync_resolve(container)
25
- return functools.partial(self._factory_provider.sync_resolve, container)
26
-
27
- @property
28
- def cast(self) -> typing.Callable[[], T_co]: # type: ignore[override]
29
- return typing.cast(typing.Callable[[], T_co], self)
File without changes
File without changes
File without changes
File without changes
File without changes