python-cq 0.2.0__tar.gz → 0.2.1__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.
- {python_cq-0.2.0 → python_cq-0.2.1}/PKG-INFO +1 -1
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/dispatcher/base.py +3 -6
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/dispatcher/bus.py +3 -8
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/dispatcher/pipe.py +13 -1
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/middleware.py +5 -1
- python_cq-0.2.1/cq/exceptions.py +7 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/middlewares/retry.py +2 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/pyproject.toml +1 -1
- {python_cq-0.2.0 → python_cq-0.2.1}/README.md +0 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/__init__.py +0 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/__init__.py +0 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/command.py +0 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/dispatcher/__init__.py +0 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/dto.py +0 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/event.py +0 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/_core/query.py +0 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/middlewares/__init__.py +0 -0
- {python_cq-0.2.0 → python_cq-0.2.1}/cq/py.typed +0 -0
|
@@ -15,7 +15,7 @@ class Dispatcher[I, O](Protocol):
|
|
|
15
15
|
raise NotImplementedError
|
|
16
16
|
|
|
17
17
|
@abstractmethod
|
|
18
|
-
def dispatch_no_wait(self,
|
|
18
|
+
def dispatch_no_wait(self, *input_values: I) -> None:
|
|
19
19
|
raise NotImplementedError
|
|
20
20
|
|
|
21
21
|
@abstractmethod
|
|
@@ -31,12 +31,9 @@ class BaseDispatcher[I, O](Dispatcher[I, O], ABC):
|
|
|
31
31
|
def __init__(self) -> None:
|
|
32
32
|
self.__middleware_group = MiddlewareGroup()
|
|
33
33
|
|
|
34
|
-
def dispatch_no_wait(self,
|
|
34
|
+
def dispatch_no_wait(self, *input_values: I) -> None:
|
|
35
35
|
asyncio.gather(
|
|
36
|
-
*(
|
|
37
|
-
self.dispatch(input_value)
|
|
38
|
-
for input_value in (first_input_value, *input_values)
|
|
39
|
-
),
|
|
36
|
+
*(self.dispatch(input_value) for input_value in input_values),
|
|
40
37
|
return_exceptions=True,
|
|
41
38
|
)
|
|
42
39
|
|
|
@@ -40,13 +40,8 @@ class SubscriberDecorator[I, O]:
|
|
|
40
40
|
bus_type: BusType[I, O] | TypeAliasType | GenericAlias
|
|
41
41
|
injection_module: injection.Module = field(default_factory=injection.mod)
|
|
42
42
|
|
|
43
|
-
def __call__[
|
|
44
|
-
|
|
45
|
-
first_input_type: type[I],
|
|
46
|
-
/,
|
|
47
|
-
*input_types: type[I],
|
|
48
|
-
) -> Callable[[T], T]:
|
|
49
|
-
def decorator(wrapped: T) -> T:
|
|
43
|
+
def __call__(self, first_input_type: type[I], /, *input_types: type[I]): # type: ignore[no-untyped-def]
|
|
44
|
+
def decorator(wrapped): # type: ignore[no-untyped-def]
|
|
50
45
|
if not isclass(wrapped) or not issubclass(wrapped, Handler):
|
|
51
46
|
raise TypeError(f"`{wrapped}` isn't a valid handler.")
|
|
52
47
|
|
|
@@ -56,7 +51,7 @@ class SubscriberDecorator[I, O]:
|
|
|
56
51
|
for input_type in (first_input_type, *input_types):
|
|
57
52
|
bus.subscribe(input_type, factory)
|
|
58
53
|
|
|
59
|
-
return wrapped
|
|
54
|
+
return wrapped
|
|
60
55
|
|
|
61
56
|
return decorator
|
|
62
57
|
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
from collections.abc import Callable
|
|
2
2
|
from dataclasses import dataclass, field
|
|
3
|
-
from typing import Any, Awaitable
|
|
3
|
+
from typing import Any, Awaitable, Self
|
|
4
4
|
|
|
5
5
|
from cq._core.dispatcher.base import BaseDispatcher, Dispatcher
|
|
6
6
|
|
|
@@ -38,6 +38,18 @@ class Pipe[I, O](BaseDispatcher[I, O]):
|
|
|
38
38
|
|
|
39
39
|
return decorator(wrapped) if wrapped else decorator
|
|
40
40
|
|
|
41
|
+
def add_static_step[T](
|
|
42
|
+
self,
|
|
43
|
+
input_value: T,
|
|
44
|
+
*,
|
|
45
|
+
dispatcher: Dispatcher[T, Any] | None = None,
|
|
46
|
+
) -> Self:
|
|
47
|
+
@self.step(dispatcher=dispatcher)
|
|
48
|
+
async def converter(_: Any) -> T:
|
|
49
|
+
return input_value
|
|
50
|
+
|
|
51
|
+
return self
|
|
52
|
+
|
|
41
53
|
async def dispatch(self, input_value: I, /) -> O:
|
|
42
54
|
return await self._invoke_with_middlewares(self.__execute, input_value)
|
|
43
55
|
|
|
@@ -2,6 +2,8 @@ from collections.abc import AsyncGenerator, Awaitable, Callable, Iterator
|
|
|
2
2
|
from dataclasses import dataclass, field
|
|
3
3
|
from typing import Self
|
|
4
4
|
|
|
5
|
+
from cq.exceptions import MiddlewareError
|
|
6
|
+
|
|
5
7
|
type MiddlewareResult[T] = AsyncGenerator[None, T]
|
|
6
8
|
type Middleware[**P, T] = Callable[P, MiddlewareResult[T]]
|
|
7
9
|
|
|
@@ -47,7 +49,9 @@ class MiddlewareGroup[**P, T]:
|
|
|
47
49
|
await generator.athrow(exc)
|
|
48
50
|
else:
|
|
49
51
|
await generator.asend(value)
|
|
50
|
-
|
|
52
|
+
raise MiddlewareError(
|
|
53
|
+
f"Too many `yield` keywords in `{middleware}`."
|
|
54
|
+
)
|
|
51
55
|
|
|
52
56
|
except StopAsyncIteration:
|
|
53
57
|
...
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|