telegrinder 0.3.1__py3-none-any.whl → 0.3.2__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 (164) 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 +11 -11
  10. telegrinder/bot/cute_types/base.py +258 -234
  11. telegrinder/bot/cute_types/callback_query.py +382 -382
  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 +53 -53
  15. telegrinder/bot/cute_types/message.py +2631 -2631
  16. telegrinder/bot/cute_types/update.py +75 -75
  17. telegrinder/bot/cute_types/utils.py +92 -92
  18. telegrinder/bot/dispatch/__init__.py +55 -55
  19. telegrinder/bot/dispatch/abc.py +77 -77
  20. telegrinder/bot/dispatch/context.py +92 -92
  21. telegrinder/bot/dispatch/dispatch.py +202 -201
  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 +128 -123
  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 +16 -16
  35. telegrinder/bot/dispatch/process.py +132 -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 -211
  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 -118
  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 +57 -57
  55. telegrinder/bot/dispatch/waiter_machine/hasher/hasher.py +57 -57
  56. telegrinder/bot/dispatch/waiter_machine/hasher/message.py +53 -53
  57. telegrinder/bot/dispatch/waiter_machine/hasher/state.py +19 -19
  58. telegrinder/bot/dispatch/waiter_machine/machine.py +168 -170
  59. telegrinder/bot/dispatch/waiter_machine/middleware.py +89 -89
  60. telegrinder/bot/dispatch/waiter_machine/short_state.py +65 -65
  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 +238 -238
  66. telegrinder/bot/rules/adapter/__init__.py +9 -9
  67. telegrinder/bot/rules/adapter/abc.py +29 -29
  68. telegrinder/bot/rules/adapter/errors.py +5 -5
  69. telegrinder/bot/rules/adapter/event.py +76 -76
  70. telegrinder/bot/rules/adapter/node.py +48 -48
  71. telegrinder/bot/rules/adapter/raw_update.py +30 -30
  72. telegrinder/bot/rules/callback_data.py +171 -171
  73. telegrinder/bot/rules/chat_join.py +48 -48
  74. telegrinder/bot/rules/command.py +126 -126
  75. telegrinder/bot/rules/enum_text.py +36 -36
  76. telegrinder/bot/rules/func.py +26 -26
  77. telegrinder/bot/rules/fuzzy.py +24 -24
  78. telegrinder/bot/rules/inline.py +60 -60
  79. telegrinder/bot/rules/integer.py +20 -20
  80. telegrinder/bot/rules/is_from.py +146 -146
  81. telegrinder/bot/rules/markup.py +43 -43
  82. telegrinder/bot/rules/mention.py +14 -14
  83. telegrinder/bot/rules/message.py +17 -17
  84. telegrinder/bot/rules/message_entities.py +35 -35
  85. telegrinder/bot/rules/node.py +27 -27
  86. telegrinder/bot/rules/regex.py +37 -37
  87. telegrinder/bot/rules/rule_enum.py +72 -72
  88. telegrinder/bot/rules/start.py +42 -42
  89. telegrinder/bot/rules/state.py +37 -37
  90. telegrinder/bot/rules/text.py +33 -33
  91. telegrinder/bot/rules/update.py +15 -15
  92. telegrinder/bot/scenario/__init__.py +5 -5
  93. telegrinder/bot/scenario/abc.py +19 -19
  94. telegrinder/bot/scenario/checkbox.py +167 -147
  95. telegrinder/bot/scenario/choice.py +46 -44
  96. telegrinder/client/__init__.py +4 -4
  97. telegrinder/client/abc.py +75 -75
  98. telegrinder/client/aiohttp.py +130 -130
  99. telegrinder/model.py +244 -244
  100. telegrinder/modules.py +237 -237
  101. telegrinder/msgspec_json.py +14 -14
  102. telegrinder/msgspec_utils.py +410 -410
  103. telegrinder/node/__init__.py +20 -20
  104. telegrinder/node/attachment.py +92 -92
  105. telegrinder/node/base.py +143 -144
  106. telegrinder/node/callback_query.py +14 -14
  107. telegrinder/node/command.py +33 -33
  108. telegrinder/node/composer.py +196 -184
  109. telegrinder/node/container.py +27 -27
  110. telegrinder/node/event.py +71 -73
  111. telegrinder/node/me.py +16 -16
  112. telegrinder/node/message.py +14 -14
  113. telegrinder/node/polymorphic.py +48 -52
  114. telegrinder/node/rule.py +76 -76
  115. telegrinder/node/scope.py +38 -38
  116. telegrinder/node/source.py +71 -71
  117. telegrinder/node/text.py +21 -21
  118. telegrinder/node/tools/__init__.py +3 -3
  119. telegrinder/node/tools/generator.py +40 -40
  120. telegrinder/node/update.py +15 -15
  121. telegrinder/rules.py +0 -0
  122. telegrinder/tools/__init__.py +74 -74
  123. telegrinder/tools/buttons.py +79 -79
  124. telegrinder/tools/error_handler/__init__.py +7 -7
  125. telegrinder/tools/error_handler/abc.py +33 -33
  126. telegrinder/tools/error_handler/error.py +9 -9
  127. telegrinder/tools/error_handler/error_handler.py +193 -193
  128. telegrinder/tools/formatting/__init__.py +46 -46
  129. telegrinder/tools/formatting/html.py +308 -308
  130. telegrinder/tools/formatting/links.py +33 -33
  131. telegrinder/tools/formatting/spec_html_formats.py +111 -111
  132. telegrinder/tools/functional.py +12 -12
  133. telegrinder/tools/global_context/__init__.py +7 -7
  134. telegrinder/tools/global_context/abc.py +63 -63
  135. telegrinder/tools/global_context/global_context.py +412 -412
  136. telegrinder/tools/global_context/telegrinder_ctx.py +27 -27
  137. telegrinder/tools/i18n/__init__.py +12 -12
  138. telegrinder/tools/i18n/abc.py +32 -32
  139. telegrinder/tools/i18n/middleware/__init__.py +3 -3
  140. telegrinder/tools/i18n/middleware/abc.py +25 -25
  141. telegrinder/tools/i18n/simple.py +43 -43
  142. telegrinder/tools/kb_set/__init__.py +4 -4
  143. telegrinder/tools/kb_set/base.py +15 -15
  144. telegrinder/tools/kb_set/yaml.py +63 -63
  145. telegrinder/tools/keyboard.py +128 -128
  146. telegrinder/tools/limited_dict.py +37 -37
  147. telegrinder/tools/loop_wrapper/__init__.py +4 -4
  148. telegrinder/tools/loop_wrapper/abc.py +15 -15
  149. telegrinder/tools/loop_wrapper/loop_wrapper.py +216 -216
  150. telegrinder/tools/magic.py +168 -168
  151. telegrinder/tools/parse_mode.py +6 -6
  152. telegrinder/tools/state_storage/__init__.py +4 -4
  153. telegrinder/tools/state_storage/abc.py +35 -35
  154. telegrinder/tools/state_storage/memory.py +25 -25
  155. telegrinder/types/__init__.py +6 -6
  156. telegrinder/types/enums.py +672 -672
  157. telegrinder/types/methods.py +4633 -4633
  158. telegrinder/types/objects.py +6317 -6317
  159. telegrinder/verification_utils.py +32 -32
  160. {telegrinder-0.3.1.dist-info → telegrinder-0.3.2.dist-info}/LICENSE +22 -22
  161. {telegrinder-0.3.1.dist-info → telegrinder-0.3.2.dist-info}/METADATA +1 -1
  162. telegrinder-0.3.2.dist-info/RECORD +164 -0
  163. telegrinder-0.3.1.dist-info/RECORD +0 -164
  164. {telegrinder-0.3.1.dist-info → telegrinder-0.3.2.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,170 +1,168 @@
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
- T = typing.TypeVar("T")
20
- HasherData = typing.TypeVar("HasherData")
21
-
22
-
23
- Storage: typing.TypeAlias = dict[
24
- Hasher[EventModel, HasherData], LimitedDict[typing.Hashable, ShortState[EventModel]]
25
- ]
26
-
27
- WEEK: typing.Final[datetime.timedelta] = datetime.timedelta(days=7)
28
-
29
-
30
- class WaiterMachine:
31
- def __init__(
32
- self,
33
- dispatch: ABCDispatch | None = None,
34
- *,
35
- max_storage_size: int = 1000,
36
- base_state_lifetime: datetime.timedelta = WEEK,
37
- ) -> None:
38
- self.dispatch = dispatch
39
- self.max_storage_size = max_storage_size
40
- self.base_state_lifetime = base_state_lifetime
41
- self.storage: Storage = {}
42
-
43
- def __repr__(self) -> str:
44
- return "<{}: max_storage_size={}, base_state_lifetime={!r}>".format(
45
- self.__class__.__name__,
46
- self.max_storage_size,
47
- self.base_state_lifetime,
48
- )
49
-
50
- async def drop_all(self) -> None:
51
- """Drops all waiters in storage."""
52
-
53
- for hasher in self.storage:
54
- for ident, short_state in self.storage[hasher].items():
55
- if short_state.context:
56
- await self.drop(
57
- hasher,
58
- ident,
59
- )
60
- else:
61
- short_state.cancel()
62
-
63
- async def drop(
64
- self,
65
- hasher: Hasher[EventModel, HasherData],
66
- id: HasherData,
67
- **context: typing.Any,
68
- ) -> None:
69
- if hasher not in self.storage:
70
- raise LookupError("No record of hasher {!r} found.".format(hasher))
71
-
72
- waiter_id: typing.Hashable = hasher.get_hash_from_data(id).expect(
73
- RuntimeError("Couldn't create hash from data")
74
- )
75
- short_state = self.storage[hasher].pop(waiter_id, None)
76
- if short_state is None:
77
- raise LookupError(
78
- "Waiter with identificator {} is not found for hasher {!r}.".format(waiter_id, hasher)
79
- )
80
-
81
- if on_drop := short_state.actions.get("on_drop"):
82
- on_drop(short_state, **context)
83
-
84
- short_state.cancel()
85
-
86
- async def wait_from_event(
87
- self,
88
- view: BaseStateView[EventModel],
89
- event: EventModel,
90
- *,
91
- filter: ABCRule | None = None,
92
- release: ABCRule | None = None,
93
- lifetime: datetime.timedelta | float | None = None,
94
- **actions: typing.Unpack[WaiterActions[EventModel]],
95
- ) -> ShortStateContext[EventModel]:
96
- hasher = StateViewHasher(view)
97
- return await self.wait(
98
- hasher=hasher,
99
- data=hasher.get_data_from_event(event).expect(
100
- RuntimeError("Hasher couldn't create data from event."),
101
- ),
102
- filter=filter,
103
- release=release,
104
- lifetime=lifetime,
105
- **actions,
106
- )
107
-
108
- async def wait(
109
- self,
110
- hasher: Hasher[EventModel, HasherData],
111
- data: HasherData,
112
- *,
113
- filter: ABCRule | None = None,
114
- release: ABCRule | None = None,
115
- lifetime: datetime.timedelta | float | None = None,
116
- **actions: typing.Unpack[WaiterActions[EventModel]],
117
- ) -> ShortStateContext[EventModel]:
118
- if isinstance(lifetime, int | float):
119
- lifetime = datetime.timedelta(seconds=lifetime)
120
-
121
- event = asyncio.Event()
122
- short_state = ShortState[EventModel](
123
- event,
124
- actions,
125
- release=release,
126
- filter=filter,
127
- lifetime=lifetime or self.base_state_lifetime,
128
- )
129
- waiter_hash = hasher.get_hash_from_data(data).expect(RuntimeError("Hasher couldn't create hash."))
130
-
131
- if hasher not in self.storage:
132
- if self.dispatch:
133
- view: BaseView[EventModel] = self.dispatch.get_view(hasher.view_class).expect(
134
- RuntimeError(f"View {hasher.view_class.__name__!r} is not defined in dispatch.")
135
- )
136
- view.middlewares.insert(0, WaiterMiddleware(self, hasher))
137
- self.storage[hasher] = LimitedDict(maxlimit=self.max_storage_size)
138
-
139
- if (deleted_short_state := self.storage[hasher].set(waiter_hash, short_state)) is not None:
140
- deleted_short_state.cancel()
141
-
142
- await event.wait()
143
- self.storage[hasher].pop(waiter_hash, None)
144
- assert short_state.context is not None
145
- return short_state.context
146
-
147
- async def clear_storage(self) -> None:
148
- """Clears storage."""
149
-
150
- for hasher in self.storage:
151
- now = datetime.datetime.now()
152
- for ident, short_state in self.storage.get(hasher, {}).copy().items():
153
- if short_state.expiration_date is not None and now > short_state.expiration_date:
154
- await self.drop(
155
- hasher,
156
- ident,
157
- force=True,
158
- )
159
-
160
-
161
- async def clear_wm_storage_worker(
162
- wm: WaiterMachine,
163
- interval_seconds: int = 60,
164
- ) -> typing.NoReturn:
165
- while True:
166
- await wm.clear_storage()
167
- await asyncio.sleep(interval_seconds)
168
-
169
-
170
- __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
+ async def drop_all(self) -> None:
49
+ """Drops all waiters in storage."""
50
+
51
+ for hasher in self.storage:
52
+ for ident, short_state in self.storage[hasher].items():
53
+ if short_state.context:
54
+ await self.drop(
55
+ hasher,
56
+ ident,
57
+ )
58
+ else:
59
+ short_state.cancel()
60
+
61
+ async def drop(
62
+ self,
63
+ hasher: Hasher[EventModel, HasherData],
64
+ id: HasherData,
65
+ **context: typing.Any,
66
+ ) -> None:
67
+ if hasher not in self.storage:
68
+ raise LookupError("No record of hasher {!r} found.".format(hasher))
69
+
70
+ waiter_id: typing.Hashable = hasher.get_hash_from_data(id).expect(
71
+ RuntimeError("Couldn't create hash from data")
72
+ )
73
+ short_state = self.storage[hasher].pop(waiter_id, None)
74
+ if short_state is None:
75
+ raise LookupError(
76
+ "Waiter with identificator {} is not found for hasher {!r}.".format(waiter_id, hasher)
77
+ )
78
+
79
+ if on_drop := short_state.actions.get("on_drop"):
80
+ on_drop(short_state, **context)
81
+
82
+ short_state.cancel()
83
+
84
+ async def wait_from_event(
85
+ self,
86
+ view: BaseStateView[EventModel],
87
+ event: EventModel,
88
+ *,
89
+ filter: ABCRule | None = None,
90
+ release: ABCRule | None = None,
91
+ lifetime: datetime.timedelta | float | None = None,
92
+ **actions: typing.Unpack[WaiterActions[EventModel]],
93
+ ) -> ShortStateContext[EventModel]:
94
+ hasher = StateViewHasher(view)
95
+ return await self.wait(
96
+ hasher=hasher,
97
+ data=hasher.get_data_from_event(event).expect(
98
+ RuntimeError("Hasher couldn't create data from event."),
99
+ ),
100
+ filter=filter,
101
+ release=release,
102
+ lifetime=lifetime,
103
+ **actions,
104
+ )
105
+
106
+ async def wait(
107
+ self,
108
+ hasher: Hasher[EventModel, HasherData],
109
+ data: HasherData,
110
+ *,
111
+ filter: ABCRule | None = None,
112
+ release: ABCRule | None = None,
113
+ lifetime: datetime.timedelta | float | None = None,
114
+ **actions: typing.Unpack[WaiterActions[EventModel]],
115
+ ) -> ShortStateContext[EventModel]:
116
+ if isinstance(lifetime, int | float):
117
+ lifetime = datetime.timedelta(seconds=lifetime)
118
+
119
+ event = asyncio.Event()
120
+ short_state = ShortState[EventModel](
121
+ event,
122
+ actions,
123
+ release=release,
124
+ filter=filter,
125
+ lifetime=lifetime or self.base_state_lifetime,
126
+ )
127
+ waiter_hash = hasher.get_hash_from_data(data).expect(RuntimeError("Hasher couldn't create hash."))
128
+
129
+ if hasher not in self.storage:
130
+ if self.dispatch:
131
+ view: BaseView[EventModel] = self.dispatch.get_view(hasher.view_class).expect(
132
+ RuntimeError(f"View {hasher.view_class.__name__!r} is not defined in dispatch.")
133
+ )
134
+ view.middlewares.insert(0, WaiterMiddleware(self, hasher))
135
+ self.storage[hasher] = LimitedDict(maxlimit=self.max_storage_size)
136
+
137
+ if (deleted_short_state := self.storage[hasher].set(waiter_hash, short_state)) is not None:
138
+ deleted_short_state.cancel()
139
+
140
+ await event.wait()
141
+ self.storage[hasher].pop(waiter_hash, None)
142
+ assert short_state.context is not None
143
+ return short_state.context
144
+
145
+ async def clear_storage(self) -> None:
146
+ """Clears storage."""
147
+
148
+ for hasher in self.storage:
149
+ now = datetime.datetime.now()
150
+ for ident, short_state in self.storage.get(hasher, {}).copy().items():
151
+ if short_state.expiration_date is not None and now > short_state.expiration_date:
152
+ await self.drop(
153
+ hasher,
154
+ ident,
155
+ force=True,
156
+ )
157
+
158
+
159
+ async def clear_wm_storage_worker(
160
+ wm: WaiterMachine,
161
+ interval_seconds: int = 60,
162
+ ) -> typing.NoReturn:
163
+ while True:
164
+ await wm.clear_storage()
165
+ await asyncio.sleep(interval_seconds)
166
+
167
+
168
+ __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",)