telegrinder 0.3.4__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 -104
  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 -22
  35. telegrinder/bot/dispatch/process.py +157 -157
  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 -172
  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 +206 -206
  66. telegrinder/bot/rules/adapter/__init__.py +17 -17
  67. telegrinder/bot/rules/adapter/abc.py +31 -31
  68. telegrinder/bot/rules/adapter/errors.py +5 -5
  69. telegrinder/bot/rules/adapter/event.py +65 -65
  70. telegrinder/bot/rules/adapter/node.py +48 -48
  71. telegrinder/bot/rules/adapter/raw_event.py +27 -27
  72. telegrinder/bot/rules/adapter/raw_update.py +30 -30
  73. telegrinder/bot/rules/callback_data.py +163 -163
  74. telegrinder/bot/rules/chat_join.py +43 -43
  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 +56 -56
  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 -176
  96. telegrinder/bot/scenario/choice.py +51 -51
  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 +313 -313
  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 +20 -20
  105. telegrinder/node/attachment.py +87 -87
  106. telegrinder/node/base.py +157 -157
  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 +5 -5
  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 +128 -128
  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 -6950
  160. telegrinder/verification_utils.py +32 -32
  161. {telegrinder-0.3.4.dist-info → telegrinder-0.3.4.post1.dist-info}/LICENSE +22 -22
  162. {telegrinder-0.3.4.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.4.dist-info/RECORD +0 -165
  165. {telegrinder-0.3.4.dist-info → telegrinder-0.3.4.post1.dist-info}/WHEEL +0 -0
@@ -1,19 +1,19 @@
1
- from fntypes.option import Option
2
-
3
- from telegrinder.bot.dispatch.view import BaseStateView
4
- from telegrinder.bot.dispatch.waiter_machine.hasher.hasher import ECHO, Event, Hasher
5
- from telegrinder.tools.functional import from_optional
6
-
7
-
8
- class StateViewHasher(Hasher[Event, int]):
9
- view: BaseStateView[Event]
10
-
11
- def __init__(self, view: BaseStateView[Event]) -> None:
12
- self.view = view
13
- super().__init__(view.__class__, get_hash_from_data=ECHO)
14
-
15
- def get_data_from_event(self, event: Event) -> Option[int]:
16
- return from_optional(self.view.get_state_key(event))
17
-
18
-
19
- __all__ = ("StateViewHasher",)
1
+ from fntypes.option import Option
2
+
3
+ from telegrinder.bot.dispatch.view import BaseStateView
4
+ from telegrinder.bot.dispatch.waiter_machine.hasher.hasher import ECHO, Event, Hasher
5
+ from telegrinder.tools.functional import from_optional
6
+
7
+
8
+ class StateViewHasher(Hasher[Event, int]):
9
+ view: BaseStateView[Event]
10
+
11
+ def __init__(self, view: BaseStateView[Event]) -> None:
12
+ self.view = view
13
+ super().__init__(view.__class__, get_hash_from_data=ECHO)
14
+
15
+ def get_data_from_event(self, event: Event) -> Option[int]:
16
+ return from_optional(self.view.get_state_key(event))
17
+
18
+
19
+ __all__ = ("StateViewHasher",)
@@ -1,172 +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
- 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
+ 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",)