telegrinder 0.3.3.post1__py3-none-any.whl → 0.3.4.post1__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 telegrinder might be problematic. Click here for more details.

Files changed (165) hide show
  1. telegrinder/__init__.py +144 -144
  2. telegrinder/api/__init__.py +8 -8
  3. telegrinder/api/api.py +93 -93
  4. telegrinder/api/error.py +16 -16
  5. telegrinder/api/response.py +20 -20
  6. telegrinder/api/token.py +36 -36
  7. telegrinder/bot/__init__.py +66 -66
  8. telegrinder/bot/bot.py +76 -76
  9. telegrinder/bot/cute_types/__init__.py +17 -17
  10. telegrinder/bot/cute_types/base.py +258 -258
  11. telegrinder/bot/cute_types/callback_query.py +385 -385
  12. telegrinder/bot/cute_types/chat_join_request.py +61 -61
  13. telegrinder/bot/cute_types/chat_member_updated.py +160 -160
  14. telegrinder/bot/cute_types/inline_query.py +43 -43
  15. telegrinder/bot/cute_types/message.py +2637 -2637
  16. telegrinder/bot/cute_types/update.py +104 -109
  17. telegrinder/bot/cute_types/utils.py +95 -95
  18. telegrinder/bot/dispatch/__init__.py +55 -55
  19. telegrinder/bot/dispatch/abc.py +77 -77
  20. telegrinder/bot/dispatch/context.py +98 -98
  21. telegrinder/bot/dispatch/dispatch.py +202 -202
  22. telegrinder/bot/dispatch/handler/__init__.py +13 -13
  23. telegrinder/bot/dispatch/handler/abc.py +24 -24
  24. telegrinder/bot/dispatch/handler/audio_reply.py +44 -44
  25. telegrinder/bot/dispatch/handler/base.py +57 -57
  26. telegrinder/bot/dispatch/handler/document_reply.py +44 -44
  27. telegrinder/bot/dispatch/handler/func.py +135 -135
  28. telegrinder/bot/dispatch/handler/media_group_reply.py +43 -43
  29. telegrinder/bot/dispatch/handler/message_reply.py +36 -36
  30. telegrinder/bot/dispatch/handler/photo_reply.py +44 -44
  31. telegrinder/bot/dispatch/handler/sticker_reply.py +37 -37
  32. telegrinder/bot/dispatch/handler/video_reply.py +44 -44
  33. telegrinder/bot/dispatch/middleware/__init__.py +3 -3
  34. telegrinder/bot/dispatch/middleware/abc.py +22 -16
  35. telegrinder/bot/dispatch/process.py +157 -132
  36. telegrinder/bot/dispatch/return_manager/__init__.py +13 -13
  37. telegrinder/bot/dispatch/return_manager/abc.py +108 -108
  38. telegrinder/bot/dispatch/return_manager/callback_query.py +20 -20
  39. telegrinder/bot/dispatch/return_manager/inline_query.py +15 -15
  40. telegrinder/bot/dispatch/return_manager/message.py +36 -36
  41. telegrinder/bot/dispatch/view/__init__.py +13 -13
  42. telegrinder/bot/dispatch/view/abc.py +41 -41
  43. telegrinder/bot/dispatch/view/base.py +200 -200
  44. telegrinder/bot/dispatch/view/box.py +129 -129
  45. telegrinder/bot/dispatch/view/callback_query.py +17 -17
  46. telegrinder/bot/dispatch/view/chat_join_request.py +16 -16
  47. telegrinder/bot/dispatch/view/chat_member.py +39 -39
  48. telegrinder/bot/dispatch/view/inline_query.py +17 -17
  49. telegrinder/bot/dispatch/view/message.py +44 -44
  50. telegrinder/bot/dispatch/view/raw.py +114 -114
  51. telegrinder/bot/dispatch/waiter_machine/__init__.py +17 -17
  52. telegrinder/bot/dispatch/waiter_machine/actions.py +13 -13
  53. telegrinder/bot/dispatch/waiter_machine/hasher/__init__.py +8 -8
  54. telegrinder/bot/dispatch/waiter_machine/hasher/callback.py +55 -55
  55. telegrinder/bot/dispatch/waiter_machine/hasher/hasher.py +57 -57
  56. telegrinder/bot/dispatch/waiter_machine/hasher/message.py +51 -51
  57. telegrinder/bot/dispatch/waiter_machine/hasher/state.py +19 -19
  58. telegrinder/bot/dispatch/waiter_machine/machine.py +172 -167
  59. telegrinder/bot/dispatch/waiter_machine/middleware.py +89 -89
  60. telegrinder/bot/dispatch/waiter_machine/short_state.py +68 -68
  61. telegrinder/bot/polling/__init__.py +4 -4
  62. telegrinder/bot/polling/abc.py +25 -25
  63. telegrinder/bot/polling/polling.py +131 -131
  64. telegrinder/bot/rules/__init__.py +62 -62
  65. telegrinder/bot/rules/abc.py +213 -213
  66. telegrinder/bot/rules/adapter/__init__.py +12 -9
  67. telegrinder/bot/rules/adapter/abc.py +31 -29
  68. telegrinder/bot/rules/adapter/errors.py +5 -5
  69. telegrinder/bot/rules/adapter/event.py +65 -67
  70. telegrinder/bot/rules/adapter/node.py +48 -48
  71. telegrinder/bot/rules/adapter/raw_event.py +27 -0
  72. telegrinder/bot/rules/adapter/raw_update.py +30 -30
  73. telegrinder/bot/rules/callback_data.py +170 -170
  74. telegrinder/bot/rules/chat_join.py +46 -46
  75. telegrinder/bot/rules/command.py +126 -126
  76. telegrinder/bot/rules/enum_text.py +36 -36
  77. telegrinder/bot/rules/func.py +26 -26
  78. telegrinder/bot/rules/fuzzy.py +24 -24
  79. telegrinder/bot/rules/inline.py +60 -60
  80. telegrinder/bot/rules/integer.py +20 -20
  81. telegrinder/bot/rules/is_from.py +127 -127
  82. telegrinder/bot/rules/markup.py +43 -43
  83. telegrinder/bot/rules/mention.py +14 -14
  84. telegrinder/bot/rules/message.py +17 -17
  85. telegrinder/bot/rules/message_entities.py +35 -35
  86. telegrinder/bot/rules/node.py +27 -27
  87. telegrinder/bot/rules/regex.py +37 -37
  88. telegrinder/bot/rules/rule_enum.py +72 -72
  89. telegrinder/bot/rules/start.py +42 -42
  90. telegrinder/bot/rules/state.py +37 -37
  91. telegrinder/bot/rules/text.py +33 -33
  92. telegrinder/bot/rules/update.py +15 -15
  93. telegrinder/bot/scenario/__init__.py +5 -5
  94. telegrinder/bot/scenario/abc.py +19 -19
  95. telegrinder/bot/scenario/checkbox.py +176 -167
  96. telegrinder/bot/scenario/choice.py +51 -46
  97. telegrinder/client/__init__.py +4 -4
  98. telegrinder/client/abc.py +75 -75
  99. telegrinder/client/aiohttp.py +130 -130
  100. telegrinder/model.py +320 -295
  101. telegrinder/modules.py +237 -237
  102. telegrinder/msgspec_json.py +14 -14
  103. telegrinder/msgspec_utils.py +410 -410
  104. telegrinder/node/__init__.py +0 -0
  105. telegrinder/node/attachment.py +87 -87
  106. telegrinder/node/base.py +166 -166
  107. telegrinder/node/callback_query.py +53 -53
  108. telegrinder/node/command.py +33 -33
  109. telegrinder/node/composer.py +198 -198
  110. telegrinder/node/container.py +27 -27
  111. telegrinder/node/event.py +65 -65
  112. telegrinder/node/me.py +16 -16
  113. telegrinder/node/message.py +14 -14
  114. telegrinder/node/polymorphic.py +48 -48
  115. telegrinder/node/rule.py +76 -76
  116. telegrinder/node/scope.py +38 -38
  117. telegrinder/node/source.py +71 -71
  118. telegrinder/node/text.py +41 -41
  119. telegrinder/node/tools/__init__.py +3 -3
  120. telegrinder/node/tools/generator.py +40 -40
  121. telegrinder/node/update.py +15 -15
  122. telegrinder/rules.py +0 -0
  123. telegrinder/tools/__init__.py +74 -74
  124. telegrinder/tools/buttons.py +79 -79
  125. telegrinder/tools/error_handler/__init__.py +7 -7
  126. telegrinder/tools/error_handler/abc.py +33 -33
  127. telegrinder/tools/error_handler/error.py +9 -9
  128. telegrinder/tools/error_handler/error_handler.py +193 -193
  129. telegrinder/tools/formatting/__init__.py +46 -46
  130. telegrinder/tools/formatting/html.py +283 -283
  131. telegrinder/tools/formatting/links.py +33 -33
  132. telegrinder/tools/formatting/spec_html_formats.py +111 -111
  133. telegrinder/tools/functional.py +12 -12
  134. telegrinder/tools/global_context/__init__.py +7 -7
  135. telegrinder/tools/global_context/abc.py +63 -63
  136. telegrinder/tools/global_context/global_context.py +412 -412
  137. telegrinder/tools/global_context/telegrinder_ctx.py +27 -27
  138. telegrinder/tools/i18n/__init__.py +7 -7
  139. telegrinder/tools/i18n/abc.py +30 -30
  140. telegrinder/tools/i18n/middleware/__init__.py +3 -3
  141. telegrinder/tools/i18n/middleware/abc.py +25 -25
  142. telegrinder/tools/i18n/simple.py +43 -43
  143. telegrinder/tools/kb_set/__init__.py +4 -4
  144. telegrinder/tools/kb_set/base.py +15 -15
  145. telegrinder/tools/kb_set/yaml.py +63 -63
  146. telegrinder/tools/keyboard.py +132 -132
  147. telegrinder/tools/limited_dict.py +37 -37
  148. telegrinder/tools/loop_wrapper/__init__.py +4 -4
  149. telegrinder/tools/loop_wrapper/abc.py +15 -15
  150. telegrinder/tools/loop_wrapper/loop_wrapper.py +224 -224
  151. telegrinder/tools/magic.py +157 -157
  152. telegrinder/tools/parse_mode.py +6 -6
  153. telegrinder/tools/state_storage/__init__.py +4 -4
  154. telegrinder/tools/state_storage/abc.py +35 -35
  155. telegrinder/tools/state_storage/memory.py +25 -25
  156. telegrinder/types/__init__.py +260 -260
  157. telegrinder/types/enums.py +701 -701
  158. telegrinder/types/methods.py +4633 -4633
  159. telegrinder/types/objects.py +6950 -8561
  160. telegrinder/verification_utils.py +32 -32
  161. {telegrinder-0.3.3.post1.dist-info → telegrinder-0.3.4.post1.dist-info}/LICENSE +22 -22
  162. {telegrinder-0.3.3.post1.dist-info → telegrinder-0.3.4.post1.dist-info}/METADATA +1 -1
  163. telegrinder-0.3.4.post1.dist-info/RECORD +165 -0
  164. telegrinder-0.3.3.post1.dist-info/RECORD +0 -164
  165. {telegrinder-0.3.3.post1.dist-info → telegrinder-0.3.4.post1.dist-info}/WHEEL +0 -0
@@ -1,167 +1,172 @@
1
- import asyncio
2
- import datetime
3
- import typing
4
-
5
- from telegrinder.bot.dispatch.abc import ABCDispatch
6
- from telegrinder.bot.dispatch.view.base import BaseStateView, BaseView
7
- from telegrinder.bot.dispatch.waiter_machine.middleware import WaiterMiddleware
8
- from telegrinder.bot.dispatch.waiter_machine.short_state import (
9
- EventModel,
10
- ShortState,
11
- ShortStateContext,
12
- )
13
- from telegrinder.bot.rules.abc import ABCRule
14
- from telegrinder.tools.limited_dict import LimitedDict
15
-
16
- from .actions import WaiterActions
17
- from .hasher import Hasher, StateViewHasher
18
-
19
- HasherData = typing.TypeVar("HasherData")
20
-
21
- Storage: typing.TypeAlias = dict[
22
- Hasher[EventModel, HasherData], LimitedDict[typing.Hashable, ShortState[EventModel]]
23
- ]
24
-
25
- WEEK: typing.Final[datetime.timedelta] = datetime.timedelta(days=7)
26
-
27
-
28
- class WaiterMachine:
29
- def __init__(
30
- self,
31
- dispatch: ABCDispatch | None = None,
32
- *,
33
- max_storage_size: int = 1000,
34
- base_state_lifetime: datetime.timedelta = WEEK,
35
- ) -> None:
36
- self.dispatch = dispatch
37
- self.max_storage_size = max_storage_size
38
- self.base_state_lifetime = base_state_lifetime
39
- self.storage: Storage = {}
40
-
41
- def __repr__(self) -> str:
42
- return "<{}: max_storage_size={}, base_state_lifetime={!r}>".format(
43
- self.__class__.__name__,
44
- self.max_storage_size,
45
- self.base_state_lifetime,
46
- )
47
-
48
- async def drop_all(self) -> None:
49
- """Drops all waiters in storage."""
50
-
51
- for hasher in self.storage.copy():
52
- for ident, short_state in self.storage[hasher].items():
53
- if short_state.context:
54
- await self.drop(hasher, ident)
55
- else:
56
- await short_state.cancel()
57
-
58
- del self.storage[hasher]
59
-
60
- async def drop(
61
- self,
62
- hasher: Hasher[EventModel, HasherData],
63
- id: HasherData,
64
- **context: typing.Any,
65
- ) -> None:
66
- if hasher not in self.storage:
67
- raise LookupError("No record of hasher {!r} found.".format(hasher))
68
-
69
- waiter_id: typing.Hashable = hasher.get_hash_from_data(id).expect(
70
- RuntimeError("Couldn't create hash from data")
71
- )
72
- short_state = self.storage[hasher].pop(waiter_id, None)
73
- if short_state is None:
74
- raise LookupError(
75
- "Waiter with identificator {} is not found for hasher {!r}.".format(waiter_id, hasher)
76
- )
77
-
78
- if on_drop := short_state.actions.get("on_drop"):
79
- on_drop(short_state, **context)
80
-
81
- await short_state.cancel()
82
-
83
- async def wait_from_event(
84
- self,
85
- view: BaseStateView[EventModel],
86
- event: EventModel,
87
- *,
88
- filter: ABCRule | None = None,
89
- release: ABCRule | None = None,
90
- lifetime: datetime.timedelta | float | None = None,
91
- **actions: typing.Unpack[WaiterActions[EventModel]],
92
- ) -> ShortStateContext[EventModel]:
93
- hasher = StateViewHasher(view)
94
- return await self.wait(
95
- hasher=hasher,
96
- data=hasher.get_data_from_event(event).expect(
97
- RuntimeError("Hasher couldn't create data from event."),
98
- ),
99
- filter=filter,
100
- release=release,
101
- lifetime=lifetime,
102
- **actions,
103
- )
104
-
105
- async def wait(
106
- self,
107
- hasher: Hasher[EventModel, HasherData],
108
- data: HasherData,
109
- *,
110
- filter: ABCRule | None = None,
111
- release: ABCRule | None = None,
112
- lifetime: datetime.timedelta | float | None = None,
113
- **actions: typing.Unpack[WaiterActions[EventModel]],
114
- ) -> ShortStateContext[EventModel]:
115
- if isinstance(lifetime, int | float):
116
- lifetime = datetime.timedelta(seconds=lifetime)
117
-
118
- event = asyncio.Event()
119
- short_state = ShortState[EventModel](
120
- event,
121
- actions,
122
- release=release,
123
- filter=filter,
124
- lifetime=lifetime or self.base_state_lifetime,
125
- )
126
- waiter_hash = hasher.get_hash_from_data(data).expect(RuntimeError("Hasher couldn't create hash."))
127
-
128
- if hasher not in self.storage:
129
- if self.dispatch:
130
- view: BaseView[EventModel] = self.dispatch.get_view(hasher.view_class).expect(
131
- RuntimeError(f"View {hasher.view_class.__name__!r} is not defined in dispatch.")
132
- )
133
- view.middlewares.insert(0, WaiterMiddleware(self, hasher))
134
- self.storage[hasher] = LimitedDict(maxlimit=self.max_storage_size)
135
-
136
- if (deleted_short_state := self.storage[hasher].set(waiter_hash, short_state)) is not None:
137
- await deleted_short_state.cancel()
138
-
139
- await event.wait()
140
- self.storage[hasher].pop(waiter_hash, None)
141
- assert short_state.context is not None
142
- return short_state.context
143
-
144
- async def clear_storage(self) -> None:
145
- """Clears storage."""
146
-
147
- for hasher in self.storage:
148
- now = datetime.datetime.now()
149
- for ident, short_state in self.storage.get(hasher, {}).copy().items():
150
- if short_state.expiration_date is not None and now > short_state.expiration_date:
151
- await self.drop(
152
- hasher,
153
- ident,
154
- force=True,
155
- )
156
-
157
-
158
- async def clear_wm_storage_worker(
159
- wm: WaiterMachine,
160
- interval_seconds: int = 60,
161
- ) -> typing.NoReturn:
162
- while True:
163
- await wm.clear_storage()
164
- await asyncio.sleep(interval_seconds)
165
-
166
-
167
- __all__ = ("WaiterMachine",)
1
+ import asyncio
2
+ import datetime
3
+ import typing
4
+
5
+ from telegrinder.bot.dispatch.abc import ABCDispatch
6
+ from telegrinder.bot.dispatch.view.base import BaseStateView, BaseView
7
+ from telegrinder.bot.dispatch.waiter_machine.middleware import WaiterMiddleware
8
+ from telegrinder.bot.dispatch.waiter_machine.short_state import (
9
+ EventModel,
10
+ ShortState,
11
+ ShortStateContext,
12
+ )
13
+ from telegrinder.bot.rules.abc import ABCRule
14
+ from telegrinder.tools.limited_dict import LimitedDict
15
+
16
+ from .actions import WaiterActions
17
+ from .hasher import Hasher, StateViewHasher
18
+
19
+ HasherData = typing.TypeVar("HasherData")
20
+
21
+ Storage: typing.TypeAlias = dict[
22
+ Hasher[EventModel, HasherData], LimitedDict[typing.Hashable, ShortState[EventModel]]
23
+ ]
24
+
25
+ WEEK: typing.Final[datetime.timedelta] = datetime.timedelta(days=7)
26
+
27
+
28
+ class WaiterMachine:
29
+ def __init__(
30
+ self,
31
+ dispatch: ABCDispatch | None = None,
32
+ *,
33
+ max_storage_size: int = 1000,
34
+ base_state_lifetime: datetime.timedelta = WEEK,
35
+ ) -> None:
36
+ self.dispatch = dispatch
37
+ self.max_storage_size = max_storage_size
38
+ self.base_state_lifetime = base_state_lifetime
39
+ self.storage: Storage = {}
40
+
41
+ def __repr__(self) -> str:
42
+ return "<{}: max_storage_size={}, base_state_lifetime={!r}>".format(
43
+ self.__class__.__name__,
44
+ self.max_storage_size,
45
+ self.base_state_lifetime,
46
+ )
47
+
48
+ def create_middleware(self, view: BaseStateView[EventModel]) -> WaiterMiddleware[EventModel]:
49
+ hasher = StateViewHasher(view)
50
+ self.storage[hasher] = LimitedDict(maxlimit=self.max_storage_size)
51
+ return WaiterMiddleware(self, hasher)
52
+
53
+ async def drop_all(self) -> None:
54
+ """Drops all waiters in storage."""
55
+
56
+ for hasher in self.storage.copy():
57
+ for ident, short_state in self.storage[hasher].items():
58
+ if short_state.context:
59
+ await self.drop(hasher, ident)
60
+ else:
61
+ await short_state.cancel()
62
+
63
+ del self.storage[hasher]
64
+
65
+ async def drop(
66
+ self,
67
+ hasher: Hasher[EventModel, HasherData],
68
+ id: HasherData,
69
+ **context: typing.Any,
70
+ ) -> None:
71
+ if hasher not in self.storage:
72
+ raise LookupError("No record of hasher {!r} found.".format(hasher))
73
+
74
+ waiter_id: typing.Hashable = hasher.get_hash_from_data(id).expect(
75
+ RuntimeError("Couldn't create hash from data")
76
+ )
77
+ short_state = self.storage[hasher].pop(waiter_id, None)
78
+ if short_state is None:
79
+ raise LookupError(
80
+ "Waiter with identificator {} is not found for hasher {!r}.".format(waiter_id, hasher)
81
+ )
82
+
83
+ if on_drop := short_state.actions.get("on_drop"):
84
+ on_drop(short_state, **context)
85
+
86
+ await short_state.cancel()
87
+
88
+ async def wait_from_event(
89
+ self,
90
+ view: BaseStateView[EventModel],
91
+ event: EventModel,
92
+ *,
93
+ filter: ABCRule | None = None,
94
+ release: ABCRule | None = None,
95
+ lifetime: datetime.timedelta | float | None = None,
96
+ **actions: typing.Unpack[WaiterActions[EventModel]],
97
+ ) -> ShortStateContext[EventModel]:
98
+ hasher = StateViewHasher(view)
99
+ return await self.wait(
100
+ hasher=hasher,
101
+ data=hasher.get_data_from_event(event).expect(
102
+ RuntimeError("Hasher couldn't create data from event."),
103
+ ),
104
+ filter=filter,
105
+ release=release,
106
+ lifetime=lifetime,
107
+ **actions,
108
+ )
109
+
110
+ async def wait(
111
+ self,
112
+ hasher: Hasher[EventModel, HasherData],
113
+ data: HasherData,
114
+ *,
115
+ filter: ABCRule | None = None,
116
+ release: ABCRule | None = None,
117
+ lifetime: datetime.timedelta | float | None = None,
118
+ **actions: typing.Unpack[WaiterActions[EventModel]],
119
+ ) -> ShortStateContext[EventModel]:
120
+ if isinstance(lifetime, int | float):
121
+ lifetime = datetime.timedelta(seconds=lifetime)
122
+
123
+ event = asyncio.Event()
124
+ short_state = ShortState[EventModel](
125
+ event,
126
+ actions,
127
+ release=release,
128
+ filter=filter,
129
+ lifetime=lifetime or self.base_state_lifetime,
130
+ )
131
+ waiter_hash = hasher.get_hash_from_data(data).expect(RuntimeError("Hasher couldn't create hash."))
132
+
133
+ if hasher not in self.storage:
134
+ if self.dispatch:
135
+ view: BaseView[EventModel] = self.dispatch.get_view(hasher.view_class).expect(
136
+ RuntimeError(f"View {hasher.view_class.__name__!r} is not defined in dispatch.")
137
+ )
138
+ view.middlewares.insert(0, WaiterMiddleware(self, hasher))
139
+ self.storage[hasher] = LimitedDict(maxlimit=self.max_storage_size)
140
+
141
+ if (deleted_short_state := self.storage[hasher].set(waiter_hash, short_state)) is not None:
142
+ await deleted_short_state.cancel()
143
+
144
+ await event.wait()
145
+ self.storage[hasher].pop(waiter_hash, None)
146
+ assert short_state.context is not None
147
+ return short_state.context
148
+
149
+ async def clear_storage(self) -> None:
150
+ """Clears storage."""
151
+
152
+ for hasher in self.storage:
153
+ now = datetime.datetime.now()
154
+ for ident, short_state in self.storage.get(hasher, {}).copy().items():
155
+ if short_state.expiration_date is not None and now > short_state.expiration_date:
156
+ await self.drop(
157
+ hasher,
158
+ ident,
159
+ force=True,
160
+ )
161
+
162
+
163
+ async def clear_wm_storage_worker(
164
+ wm: WaiterMachine,
165
+ interval_seconds: int = 60,
166
+ ) -> typing.NoReturn:
167
+ while True:
168
+ await wm.clear_storage()
169
+ await asyncio.sleep(interval_seconds)
170
+
171
+
172
+ __all__ = ("WaiterMachine",)
@@ -1,89 +1,89 @@
1
- import datetime
2
- import typing
3
-
4
- from telegrinder.bot.cute_types.base import BaseCute
5
- from telegrinder.bot.dispatch.context import Context
6
- from telegrinder.bot.dispatch.handler.func import FuncHandler
7
- from telegrinder.bot.dispatch.middleware.abc import ABCMiddleware
8
- from telegrinder.bot.dispatch.process import check_rule
9
- from telegrinder.bot.dispatch.waiter_machine.short_state import ShortStateContext
10
- from telegrinder.modules import logger
11
-
12
- from .hasher import Hasher
13
-
14
- if typing.TYPE_CHECKING:
15
- from .machine import WaiterMachine
16
- from .short_state import ShortState
17
-
18
- EventType = typing.TypeVar("EventType", bound=BaseCute)
19
-
20
-
21
- class WaiterMiddleware(ABCMiddleware[EventType]):
22
- def __init__(
23
- self,
24
- machine: "WaiterMachine",
25
- hasher: Hasher,
26
- ) -> None:
27
- self.machine = machine
28
- self.hasher = hasher
29
-
30
- async def pre(self, event: EventType, ctx: Context) -> bool:
31
- if self.hasher not in self.machine.storage:
32
- return True
33
-
34
- key = self.hasher.get_hash_from_data_from_event(event)
35
- if key is None:
36
- logger.info(f"Unable to get hash from event with hasher {self.hasher}")
37
- return True
38
-
39
- short_state: "ShortState[EventType] | None" = self.machine.storage[self.hasher].get(key.unwrap())
40
- if not short_state:
41
- return True
42
-
43
- preset_context = Context(short_state=short_state)
44
- if short_state.context is not None:
45
- preset_context.update(short_state.context.context)
46
-
47
- # Run filter rule
48
- if short_state.filter and not await check_rule(
49
- event.ctx_api, short_state.filter, ctx.raw_update, preset_context
50
- ):
51
- logger.debug("Filter rule {!r} failed", short_state.filter)
52
- return True
53
-
54
- if short_state.expiration_date is not None and datetime.datetime.now() >= short_state.expiration_date:
55
- await self.machine.drop(
56
- self.hasher,
57
- self.hasher.get_data_from_event(event).unwrap(),
58
- **preset_context.copy(),
59
- )
60
- return True
61
-
62
- handler = FuncHandler(
63
- self.pass_runtime,
64
- [short_state.release] if short_state.release else [],
65
- dataclass=None,
66
- preset_context=preset_context,
67
- )
68
- result = await handler.check(event.ctx_api, ctx.raw_update, ctx)
69
-
70
- if result is True:
71
- await handler.run(event.api, event, ctx)
72
-
73
- elif on_miss := short_state.actions.get("on_miss"): # noqa: SIM102
74
- if await on_miss.check(event.ctx_api, ctx.raw_update, ctx):
75
- await on_miss.run(event.ctx_api, event, ctx)
76
-
77
- return False
78
-
79
- async def pass_runtime(
80
- self,
81
- event: EventType,
82
- short_state: "ShortState[EventType]",
83
- ctx: Context,
84
- ) -> None:
85
- short_state.context = ShortStateContext(event, ctx)
86
- short_state.event.set()
87
-
88
-
89
- __all__ = ("WaiterMiddleware",)
1
+ import datetime
2
+ import typing
3
+
4
+ from telegrinder.bot.cute_types.base import BaseCute
5
+ from telegrinder.bot.dispatch.context import Context
6
+ from telegrinder.bot.dispatch.handler.func import FuncHandler
7
+ from telegrinder.bot.dispatch.middleware.abc import ABCMiddleware
8
+ from telegrinder.bot.dispatch.process import check_rule
9
+ from telegrinder.bot.dispatch.waiter_machine.short_state import ShortStateContext
10
+ from telegrinder.modules import logger
11
+
12
+ from .hasher import Hasher
13
+
14
+ if typing.TYPE_CHECKING:
15
+ from .machine import WaiterMachine
16
+ from .short_state import ShortState
17
+
18
+ EventType = typing.TypeVar("EventType", bound=BaseCute)
19
+
20
+
21
+ class WaiterMiddleware(ABCMiddleware[EventType]):
22
+ def __init__(
23
+ self,
24
+ machine: "WaiterMachine",
25
+ hasher: Hasher,
26
+ ) -> None:
27
+ self.machine = machine
28
+ self.hasher = hasher
29
+
30
+ async def pre(self, event: EventType, ctx: Context) -> bool:
31
+ if self.hasher not in self.machine.storage:
32
+ return True
33
+
34
+ key = self.hasher.get_hash_from_data_from_event(event)
35
+ if key is None:
36
+ logger.info(f"Unable to get hash from event with hasher {self.hasher}")
37
+ return True
38
+
39
+ short_state: "ShortState[EventType] | None" = self.machine.storage[self.hasher].get(key.unwrap())
40
+ if not short_state:
41
+ return True
42
+
43
+ preset_context = Context(short_state=short_state)
44
+ if short_state.context is not None:
45
+ preset_context.update(short_state.context.context)
46
+
47
+ # Run filter rule
48
+ if short_state.filter and not await check_rule(
49
+ event.ctx_api, short_state.filter, ctx.raw_update, preset_context
50
+ ):
51
+ logger.debug("Filter rule {!r} failed", short_state.filter)
52
+ return True
53
+
54
+ if short_state.expiration_date is not None and datetime.datetime.now() >= short_state.expiration_date:
55
+ await self.machine.drop(
56
+ self.hasher,
57
+ self.hasher.get_data_from_event(event).unwrap(),
58
+ **preset_context.copy(),
59
+ )
60
+ return True
61
+
62
+ handler = FuncHandler(
63
+ self.pass_runtime,
64
+ [short_state.release] if short_state.release else [],
65
+ dataclass=None,
66
+ preset_context=preset_context,
67
+ )
68
+ result = await handler.check(event.ctx_api, ctx.raw_update, ctx)
69
+
70
+ if result is True:
71
+ await handler.run(event.api, event, ctx)
72
+
73
+ elif on_miss := short_state.actions.get("on_miss"): # noqa: SIM102
74
+ if await on_miss.check(event.ctx_api, ctx.raw_update, ctx):
75
+ await on_miss.run(event.ctx_api, event, ctx)
76
+
77
+ return False
78
+
79
+ async def pass_runtime(
80
+ self,
81
+ event: EventType,
82
+ short_state: "ShortState[EventType]",
83
+ ctx: Context,
84
+ ) -> None:
85
+ short_state.context = ShortStateContext(event, ctx)
86
+ short_state.event.set()
87
+
88
+
89
+ __all__ = ("WaiterMiddleware",)
@@ -1,68 +1,68 @@
1
- import asyncio
2
- import dataclasses
3
- import datetime
4
- import typing
5
- from contextlib import suppress
6
-
7
- from telegrinder.bot.cute_types import BaseCute
8
- from telegrinder.bot.dispatch.context import Context
9
- from telegrinder.bot.dispatch.handler.abc import ABCHandler
10
- from telegrinder.bot.rules.abc import ABCRule
11
- from telegrinder.model import Model
12
-
13
- if typing.TYPE_CHECKING:
14
- from .actions import WaiterActions
15
-
16
-
17
- T = typing.TypeVar("T", bound=Model)
18
- EventModel = typing.TypeVar("EventModel", bound=BaseCute)
19
-
20
- Behaviour: typing.TypeAlias = ABCHandler[T] | None
21
-
22
-
23
- class ShortStateContext(typing.Generic[EventModel], typing.NamedTuple):
24
- event: EventModel
25
- context: Context
26
-
27
-
28
- @dataclasses.dataclass(slots=True)
29
- class ShortState(typing.Generic[EventModel]):
30
- event: asyncio.Event
31
- actions: "WaiterActions[EventModel]"
32
-
33
- release: ABCRule | None = dataclasses.field(
34
- default=None,
35
- kw_only=True,
36
- )
37
- filter: ABCRule | None = dataclasses.field(
38
- default=None,
39
- kw_only=True,
40
- )
41
-
42
- lifetime: dataclasses.InitVar[datetime.timedelta | None] = dataclasses.field(
43
- default=None,
44
- kw_only=True,
45
- )
46
-
47
- expiration_date: datetime.datetime | None = dataclasses.field(init=False, kw_only=True)
48
- creation_date: datetime.datetime = dataclasses.field(init=False)
49
- context: ShortStateContext[EventModel] | None = dataclasses.field(default=None, init=False, kw_only=True)
50
-
51
- def __post_init__(self, expiration: datetime.timedelta | None = None) -> None:
52
- self.creation_date = datetime.datetime.now()
53
- self.expiration_date = (self.creation_date + expiration) if expiration is not None else None
54
-
55
- async def cancel(self) -> None:
56
- """Cancel schedule waiters."""
57
-
58
- waiters = typing.cast(
59
- typing.Iterable[asyncio.Future[typing.Any]],
60
- self.event._waiters, # type: ignore
61
- )
62
- for future in waiters:
63
- future.cancel()
64
- with suppress(asyncio.CancelledError):
65
- await future
66
-
67
-
68
- __all__ = ("ShortState", "ShortStateContext")
1
+ import asyncio
2
+ import dataclasses
3
+ import datetime
4
+ import typing
5
+ from contextlib import suppress
6
+
7
+ from telegrinder.bot.cute_types import BaseCute
8
+ from telegrinder.bot.dispatch.context import Context
9
+ from telegrinder.bot.dispatch.handler.abc import ABCHandler
10
+ from telegrinder.bot.rules.abc import ABCRule
11
+ from telegrinder.model import Model
12
+
13
+ if typing.TYPE_CHECKING:
14
+ from .actions import WaiterActions
15
+
16
+
17
+ T = typing.TypeVar("T", bound=Model)
18
+ EventModel = typing.TypeVar("EventModel", bound=BaseCute)
19
+
20
+ Behaviour: typing.TypeAlias = ABCHandler[T] | None
21
+
22
+
23
+ class ShortStateContext(typing.Generic[EventModel], typing.NamedTuple):
24
+ event: EventModel
25
+ context: Context
26
+
27
+
28
+ @dataclasses.dataclass(slots=True)
29
+ class ShortState(typing.Generic[EventModel]):
30
+ event: asyncio.Event
31
+ actions: "WaiterActions[EventModel]"
32
+
33
+ release: ABCRule | None = dataclasses.field(
34
+ default=None,
35
+ kw_only=True,
36
+ )
37
+ filter: ABCRule | None = dataclasses.field(
38
+ default=None,
39
+ kw_only=True,
40
+ )
41
+
42
+ lifetime: dataclasses.InitVar[datetime.timedelta | None] = dataclasses.field(
43
+ default=None,
44
+ kw_only=True,
45
+ )
46
+
47
+ expiration_date: datetime.datetime | None = dataclasses.field(init=False, kw_only=True)
48
+ creation_date: datetime.datetime = dataclasses.field(init=False)
49
+ context: ShortStateContext[EventModel] | None = dataclasses.field(default=None, init=False, kw_only=True)
50
+
51
+ def __post_init__(self, expiration: datetime.timedelta | None = None) -> None:
52
+ self.creation_date = datetime.datetime.now()
53
+ self.expiration_date = (self.creation_date + expiration) if expiration is not None else None
54
+
55
+ async def cancel(self) -> None:
56
+ """Cancel schedule waiters."""
57
+
58
+ waiters = typing.cast(
59
+ typing.Iterable[asyncio.Future[typing.Any]],
60
+ self.event._waiters, # type: ignore
61
+ )
62
+ for future in waiters:
63
+ future.cancel()
64
+ with suppress(asyncio.CancelledError):
65
+ await future
66
+
67
+
68
+ __all__ = ("ShortState", "ShortStateContext")