telegrinder 0.3.4__py3-none-any.whl → 0.4.0__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.
- telegrinder/__init__.py +148 -149
- telegrinder/api/__init__.py +9 -8
- telegrinder/api/api.py +101 -93
- telegrinder/api/error.py +20 -16
- telegrinder/api/response.py +20 -20
- telegrinder/api/token.py +36 -36
- telegrinder/bot/__init__.py +72 -66
- telegrinder/bot/bot.py +83 -76
- telegrinder/bot/cute_types/__init__.py +19 -17
- telegrinder/bot/cute_types/base.py +184 -258
- telegrinder/bot/cute_types/callback_query.py +400 -385
- telegrinder/bot/cute_types/chat_join_request.py +62 -61
- telegrinder/bot/cute_types/chat_member_updated.py +157 -160
- telegrinder/bot/cute_types/inline_query.py +44 -43
- telegrinder/bot/cute_types/message.py +2590 -2637
- telegrinder/bot/cute_types/pre_checkout_query.py +42 -0
- telegrinder/bot/cute_types/update.py +112 -104
- telegrinder/bot/cute_types/utils.py +62 -95
- telegrinder/bot/dispatch/__init__.py +59 -55
- telegrinder/bot/dispatch/abc.py +76 -77
- telegrinder/bot/dispatch/context.py +96 -98
- telegrinder/bot/dispatch/dispatch.py +254 -202
- telegrinder/bot/dispatch/handler/__init__.py +13 -13
- telegrinder/bot/dispatch/handler/abc.py +23 -24
- telegrinder/bot/dispatch/handler/audio_reply.py +44 -44
- telegrinder/bot/dispatch/handler/base.py +57 -57
- telegrinder/bot/dispatch/handler/document_reply.py +44 -44
- telegrinder/bot/dispatch/handler/func.py +129 -135
- telegrinder/bot/dispatch/handler/media_group_reply.py +44 -43
- telegrinder/bot/dispatch/handler/message_reply.py +36 -36
- telegrinder/bot/dispatch/handler/photo_reply.py +44 -44
- telegrinder/bot/dispatch/handler/sticker_reply.py +37 -37
- telegrinder/bot/dispatch/handler/video_reply.py +44 -44
- telegrinder/bot/dispatch/middleware/__init__.py +3 -3
- telegrinder/bot/dispatch/middleware/abc.py +97 -22
- telegrinder/bot/dispatch/middleware/global_middleware.py +70 -0
- telegrinder/bot/dispatch/process.py +151 -157
- telegrinder/bot/dispatch/return_manager/__init__.py +15 -13
- telegrinder/bot/dispatch/return_manager/abc.py +104 -108
- telegrinder/bot/dispatch/return_manager/callback_query.py +20 -20
- telegrinder/bot/dispatch/return_manager/inline_query.py +15 -15
- telegrinder/bot/dispatch/return_manager/message.py +36 -36
- telegrinder/bot/dispatch/return_manager/pre_checkout_query.py +20 -0
- telegrinder/bot/dispatch/view/__init__.py +15 -13
- telegrinder/bot/dispatch/view/abc.py +45 -41
- telegrinder/bot/dispatch/view/base.py +231 -200
- telegrinder/bot/dispatch/view/box.py +140 -129
- telegrinder/bot/dispatch/view/callback_query.py +16 -17
- telegrinder/bot/dispatch/view/chat_join_request.py +11 -16
- telegrinder/bot/dispatch/view/chat_member.py +37 -39
- telegrinder/bot/dispatch/view/inline_query.py +16 -17
- telegrinder/bot/dispatch/view/message.py +43 -44
- telegrinder/bot/dispatch/view/pre_checkout_query.py +16 -0
- telegrinder/bot/dispatch/view/raw.py +116 -114
- telegrinder/bot/dispatch/waiter_machine/__init__.py +17 -17
- telegrinder/bot/dispatch/waiter_machine/actions.py +14 -13
- telegrinder/bot/dispatch/waiter_machine/hasher/__init__.py +8 -8
- telegrinder/bot/dispatch/waiter_machine/hasher/callback.py +55 -55
- telegrinder/bot/dispatch/waiter_machine/hasher/hasher.py +59 -57
- telegrinder/bot/dispatch/waiter_machine/hasher/message.py +51 -51
- telegrinder/bot/dispatch/waiter_machine/hasher/state.py +20 -19
- telegrinder/bot/dispatch/waiter_machine/machine.py +251 -172
- telegrinder/bot/dispatch/waiter_machine/middleware.py +94 -89
- telegrinder/bot/dispatch/waiter_machine/short_state.py +57 -68
- telegrinder/bot/polling/__init__.py +4 -4
- telegrinder/bot/polling/abc.py +25 -25
- telegrinder/bot/polling/polling.py +139 -131
- telegrinder/bot/rules/__init__.py +85 -62
- telegrinder/bot/rules/abc.py +213 -206
- telegrinder/bot/rules/callback_data.py +122 -163
- telegrinder/bot/rules/chat_join.py +45 -43
- telegrinder/bot/rules/command.py +126 -126
- telegrinder/bot/rules/enum_text.py +33 -36
- telegrinder/bot/rules/func.py +28 -26
- telegrinder/bot/rules/fuzzy.py +24 -24
- telegrinder/bot/rules/id.py +24 -0
- telegrinder/bot/rules/inline.py +58 -56
- telegrinder/bot/rules/integer.py +21 -20
- telegrinder/bot/rules/is_from.py +127 -127
- telegrinder/bot/rules/logic.py +18 -0
- telegrinder/bot/rules/markup.py +42 -43
- telegrinder/bot/rules/mention.py +14 -14
- telegrinder/bot/rules/message.py +15 -17
- telegrinder/bot/rules/message_entities.py +33 -35
- telegrinder/bot/rules/node.py +33 -27
- telegrinder/bot/rules/payload.py +81 -0
- telegrinder/bot/rules/payment_invoice.py +29 -0
- telegrinder/bot/rules/regex.py +36 -37
- telegrinder/bot/rules/rule_enum.py +72 -72
- telegrinder/bot/rules/start.py +42 -42
- telegrinder/bot/rules/state.py +35 -37
- telegrinder/bot/rules/text.py +38 -33
- telegrinder/bot/rules/update.py +15 -15
- telegrinder/bot/scenario/__init__.py +5 -5
- telegrinder/bot/scenario/abc.py +17 -19
- telegrinder/bot/scenario/checkbox.py +174 -176
- telegrinder/bot/scenario/choice.py +48 -51
- telegrinder/client/__init__.py +12 -4
- telegrinder/client/abc.py +100 -75
- telegrinder/client/aiohttp.py +134 -130
- telegrinder/client/form_data.py +31 -0
- telegrinder/client/sonic.py +212 -0
- telegrinder/model.py +208 -315
- telegrinder/modules.py +239 -237
- telegrinder/msgspec_json.py +14 -14
- telegrinder/msgspec_utils.py +478 -410
- telegrinder/node/__init__.py +86 -25
- telegrinder/node/attachment.py +163 -87
- telegrinder/node/base.py +288 -160
- telegrinder/node/callback_query.py +54 -53
- telegrinder/node/command.py +34 -33
- telegrinder/node/composer.py +163 -198
- telegrinder/node/container.py +33 -27
- telegrinder/node/either.py +82 -0
- telegrinder/node/event.py +54 -65
- telegrinder/node/file.py +51 -0
- telegrinder/node/me.py +15 -16
- telegrinder/node/payload.py +78 -0
- telegrinder/node/polymorphic.py +67 -48
- telegrinder/node/rule.py +72 -76
- telegrinder/node/scope.py +36 -38
- telegrinder/node/source.py +87 -71
- telegrinder/node/text.py +53 -41
- telegrinder/node/tools/__init__.py +3 -3
- telegrinder/node/tools/generator.py +36 -40
- telegrinder/py.typed +0 -0
- telegrinder/rules.py +1 -62
- telegrinder/tools/__init__.py +152 -93
- telegrinder/tools/adapter/__init__.py +19 -0
- telegrinder/tools/adapter/abc.py +49 -0
- telegrinder/tools/adapter/dataclass.py +56 -0
- telegrinder/{bot/rules → tools}/adapter/errors.py +5 -5
- telegrinder/{bot/rules → tools}/adapter/event.py +63 -65
- telegrinder/{bot/rules → tools}/adapter/node.py +46 -48
- telegrinder/{bot/rules → tools}/adapter/raw_event.py +27 -27
- telegrinder/{bot/rules → tools}/adapter/raw_update.py +30 -30
- telegrinder/tools/buttons.py +106 -80
- telegrinder/tools/callback_data_serilization/__init__.py +5 -0
- telegrinder/tools/callback_data_serilization/abc.py +51 -0
- telegrinder/tools/callback_data_serilization/json_ser.py +60 -0
- telegrinder/tools/callback_data_serilization/msgpack_ser.py +172 -0
- telegrinder/tools/error_handler/__init__.py +7 -7
- telegrinder/tools/error_handler/abc.py +30 -33
- telegrinder/tools/error_handler/error.py +9 -9
- telegrinder/tools/error_handler/error_handler.py +179 -193
- telegrinder/tools/formatting/__init__.py +83 -63
- telegrinder/tools/formatting/deep_links.py +541 -0
- telegrinder/tools/formatting/{html.py → html_formatter.py} +266 -294
- telegrinder/tools/formatting/spec_html_formats.py +71 -117
- telegrinder/tools/functional.py +8 -12
- telegrinder/tools/global_context/__init__.py +7 -7
- telegrinder/tools/global_context/abc.py +63 -63
- telegrinder/tools/global_context/global_context.py +387 -412
- telegrinder/tools/global_context/telegrinder_ctx.py +27 -27
- telegrinder/tools/i18n/__init__.py +7 -7
- telegrinder/tools/i18n/abc.py +30 -30
- telegrinder/tools/i18n/middleware/__init__.py +3 -3
- telegrinder/tools/i18n/middleware/abc.py +22 -25
- telegrinder/tools/i18n/simple.py +43 -43
- telegrinder/tools/input_file_directory.py +30 -0
- telegrinder/tools/keyboard.py +128 -128
- telegrinder/tools/lifespan.py +105 -0
- telegrinder/tools/limited_dict.py +32 -37
- telegrinder/tools/loop_wrapper/__init__.py +4 -4
- telegrinder/tools/loop_wrapper/abc.py +20 -15
- telegrinder/tools/loop_wrapper/loop_wrapper.py +169 -224
- telegrinder/tools/magic.py +307 -157
- telegrinder/tools/parse_mode.py +6 -6
- telegrinder/tools/state_storage/__init__.py +4 -4
- telegrinder/tools/state_storage/abc.py +31 -35
- telegrinder/tools/state_storage/memory.py +25 -25
- telegrinder/tools/strings.py +13 -0
- telegrinder/types/__init__.py +268 -260
- telegrinder/types/enums.py +711 -701
- telegrinder/types/input_file.py +51 -0
- telegrinder/types/methods.py +5055 -4633
- telegrinder/types/objects.py +7058 -6950
- telegrinder/verification_utils.py +30 -32
- {telegrinder-0.3.4.dist-info → telegrinder-0.4.0.dist-info}/LICENSE +22 -22
- telegrinder-0.4.0.dist-info/METADATA +144 -0
- telegrinder-0.4.0.dist-info/RECORD +182 -0
- {telegrinder-0.3.4.dist-info → telegrinder-0.4.0.dist-info}/WHEEL +1 -1
- telegrinder/bot/rules/adapter/__init__.py +0 -17
- telegrinder/bot/rules/adapter/abc.py +0 -31
- telegrinder/node/message.py +0 -14
- telegrinder/node/update.py +0 -15
- telegrinder/tools/formatting/links.py +0 -38
- telegrinder/tools/kb_set/__init__.py +0 -4
- telegrinder/tools/kb_set/base.py +0 -15
- telegrinder/tools/kb_set/yaml.py +0 -63
- telegrinder-0.3.4.dist-info/METADATA +0 -110
- telegrinder-0.3.4.dist-info/RECORD +0 -165
|
@@ -1,308 +1,280 @@
|
|
|
1
|
-
import
|
|
2
|
-
|
|
3
|
-
import
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
from
|
|
8
|
-
|
|
9
|
-
from .
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
"
|
|
37
|
-
"
|
|
38
|
-
"
|
|
39
|
-
"
|
|
40
|
-
"
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
)
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
return fmt
|
|
56
|
-
|
|
57
|
-
def get_spec_formatter(self, value: SpecialFormat) -> typing.Callable[...,
|
|
58
|
-
return globals()[value.__formatter_name__]
|
|
59
|
-
|
|
60
|
-
def
|
|
61
|
-
if is_spec_format(value):
|
|
62
|
-
value = value.string
|
|
63
|
-
|
|
64
|
-
current_format = globals()[fmts.pop(0)](
|
|
65
|
-
str(value)
|
|
66
|
-
if isinstance(value, TagFormat)
|
|
67
|
-
else escape(FormatString(value) if not isinstance(value, str) else value)
|
|
68
|
-
)
|
|
69
|
-
for fmt in fmts:
|
|
70
|
-
current_format = globals()[fmt](current_format)
|
|
71
|
-
|
|
72
|
-
return (
|
|
73
|
-
TagFormat(
|
|
74
|
-
current_format,
|
|
75
|
-
tag=value.tag,
|
|
76
|
-
**value.data,
|
|
77
|
-
)
|
|
78
|
-
if isinstance(value, TagFormat)
|
|
79
|
-
else current_format
|
|
80
|
-
)
|
|
81
|
-
|
|
82
|
-
def format_field(self, value: typing.Any, fmt: str) ->
|
|
83
|
-
with suppress(ValueError):
|
|
84
|
-
return HTMLFormatter(
|
|
85
|
-
format(
|
|
86
|
-
(
|
|
87
|
-
value.formatting()
|
|
88
|
-
if isinstance(value, TagFormat)
|
|
89
|
-
else (
|
|
90
|
-
self.get_spec_formatter(value)(**value
|
|
91
|
-
if is_spec_format(value)
|
|
92
|
-
else value
|
|
93
|
-
)
|
|
94
|
-
),
|
|
95
|
-
fmt,
|
|
96
|
-
)
|
|
97
|
-
)
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
return
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
return
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
def
|
|
135
|
-
"""
|
|
136
|
-
return self.
|
|
137
|
-
|
|
138
|
-
def
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
return
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
get_invite_chat_link(invite_link),
|
|
266
|
-
string or f"https://t.me/joinchat/{invite_link}",
|
|
267
|
-
)
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
def resolve_domain(username: str, string: str | None = None) -> TagFormat:
|
|
271
|
-
return link(
|
|
272
|
-
get_resolve_domain_link(username),
|
|
273
|
-
string or f"t.me/{username}",
|
|
274
|
-
)
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
def underline(string: str) -> TagFormat:
|
|
278
|
-
return TagFormat(string, tag="u")
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
__all__ = (
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
|
|
3
|
+
import dataclasses
|
|
4
|
+
import html
|
|
5
|
+
import string
|
|
6
|
+
import typing
|
|
7
|
+
from contextlib import suppress
|
|
8
|
+
|
|
9
|
+
from telegrinder.tools.parse_mode import ParseMode
|
|
10
|
+
from telegrinder.types.enums import ProgrammingLanguage
|
|
11
|
+
|
|
12
|
+
from .deep_links import tg_mention_link
|
|
13
|
+
from .spec_html_formats import SpecialFormat, is_spec_format
|
|
14
|
+
|
|
15
|
+
type HTMLFormat = str | TagFormat
|
|
16
|
+
|
|
17
|
+
HTML_UNION_SPECIFIERS_SEPARATOR: typing.Final[str] = "+"
|
|
18
|
+
TAG_FORMAT: typing.Final[str] = "<{tag}{data}>{content}</{tag}>"
|
|
19
|
+
QUOT_MARK: typing.Final[str] = '"'
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
class StringFormatterProto(typing.Protocol):
|
|
23
|
+
def format_field(self, value: typing.Any, fmt: str) -> HTMLFormatter: ...
|
|
24
|
+
|
|
25
|
+
def format(self, __string: str, *args: object, **kwargs: object) -> HTMLFormatter: ...
|
|
26
|
+
|
|
27
|
+
|
|
28
|
+
class StringFormatter(string.Formatter):
|
|
29
|
+
"""String formatter, using substitutions from args and kwargs.
|
|
30
|
+
The substitutions are identified by braces ('{' and '}') with
|
|
31
|
+
specifiers: `bold`, `italic`, `underline`, `strike` etc...
|
|
32
|
+
"""
|
|
33
|
+
|
|
34
|
+
__formats__: typing.ClassVar[tuple[str, ...]] = (
|
|
35
|
+
"bold",
|
|
36
|
+
"code_inline",
|
|
37
|
+
"italic",
|
|
38
|
+
"spoiler",
|
|
39
|
+
"strike",
|
|
40
|
+
"underline",
|
|
41
|
+
)
|
|
42
|
+
|
|
43
|
+
def validate_html_format(self, value: typing.Any, fmt: str) -> HTMLFormat:
|
|
44
|
+
if not fmt:
|
|
45
|
+
raise ValueError("Formats union should be: format+format.")
|
|
46
|
+
|
|
47
|
+
if fmt not in self.__formats__:
|
|
48
|
+
raise ValueError(
|
|
49
|
+
"Unknown format {!r} for object of type {!r}.".format(
|
|
50
|
+
fmt,
|
|
51
|
+
type(value).__name__,
|
|
52
|
+
)
|
|
53
|
+
)
|
|
54
|
+
|
|
55
|
+
return fmt
|
|
56
|
+
|
|
57
|
+
def get_spec_formatter(self, value: SpecialFormat) -> typing.Callable[..., TagFormat]:
|
|
58
|
+
return globals()[value.__formatter_name__]
|
|
59
|
+
|
|
60
|
+
def make_tag_format(self, value: typing.Any, fmts: list[HTMLFormat]) -> TagFormat:
|
|
61
|
+
if is_spec_format(value):
|
|
62
|
+
value = value.string
|
|
63
|
+
|
|
64
|
+
current_format = globals()[fmts.pop(0)](
|
|
65
|
+
str(value)
|
|
66
|
+
if isinstance(value, TagFormat)
|
|
67
|
+
else escape(FormatString(value) if not isinstance(value, str) else value)
|
|
68
|
+
)
|
|
69
|
+
for fmt in fmts:
|
|
70
|
+
current_format = globals()[fmt](current_format)
|
|
71
|
+
|
|
72
|
+
return (
|
|
73
|
+
TagFormat(
|
|
74
|
+
current_format,
|
|
75
|
+
tag=value.tag,
|
|
76
|
+
**value.data,
|
|
77
|
+
)
|
|
78
|
+
if isinstance(value, TagFormat)
|
|
79
|
+
else current_format
|
|
80
|
+
)
|
|
81
|
+
|
|
82
|
+
def format_field(self, value: typing.Any, fmt: str) -> HTMLFormatter:
|
|
83
|
+
with suppress(ValueError):
|
|
84
|
+
return HTMLFormatter(
|
|
85
|
+
format(
|
|
86
|
+
(
|
|
87
|
+
value.formatting()
|
|
88
|
+
if isinstance(value, TagFormat)
|
|
89
|
+
else (
|
|
90
|
+
self.get_spec_formatter(value)(**dataclasses.asdict(value)).formatting()
|
|
91
|
+
if is_spec_format(value)
|
|
92
|
+
else value
|
|
93
|
+
)
|
|
94
|
+
),
|
|
95
|
+
fmt,
|
|
96
|
+
)
|
|
97
|
+
)
|
|
98
|
+
|
|
99
|
+
fmts = list(
|
|
100
|
+
map(
|
|
101
|
+
lambda fmt: self.validate_html_format(value, fmt),
|
|
102
|
+
fmt.split(HTML_UNION_SPECIFIERS_SEPARATOR),
|
|
103
|
+
),
|
|
104
|
+
)
|
|
105
|
+
tag_format = self.make_tag_format(value, fmts)
|
|
106
|
+
|
|
107
|
+
if is_spec_format(value):
|
|
108
|
+
value.string = tag_format
|
|
109
|
+
tag_format = self.get_spec_formatter(value)(**dataclasses.asdict(value))
|
|
110
|
+
|
|
111
|
+
return tag_format.formatting()
|
|
112
|
+
|
|
113
|
+
def format(self, __string: str, *args: object, **kwargs: object) -> HTMLFormatter:
|
|
114
|
+
return HTMLFormatter(super().format(__string, *args, **kwargs))
|
|
115
|
+
|
|
116
|
+
|
|
117
|
+
class FormatString(str):
|
|
118
|
+
STRING_FORMATTER: StringFormatterProto = StringFormatter()
|
|
119
|
+
|
|
120
|
+
def __new__(cls, string: str, /) -> typing.Self:
|
|
121
|
+
if isinstance(string, TagFormat):
|
|
122
|
+
return super().__new__(cls, string.formatting())
|
|
123
|
+
return super().__new__(cls, string)
|
|
124
|
+
|
|
125
|
+
def __add__(self, value: str) -> HTMLFormatter:
|
|
126
|
+
"""Returns self+value."""
|
|
127
|
+
return HTMLFormatter(
|
|
128
|
+
str.__add__(
|
|
129
|
+
escape(self),
|
|
130
|
+
value.formatting() if isinstance(value, TagFormat) else escape(value),
|
|
131
|
+
)
|
|
132
|
+
)
|
|
133
|
+
|
|
134
|
+
def __radd__(self, value: str) -> HTMLFormatter:
|
|
135
|
+
"""Returns value+self."""
|
|
136
|
+
return HTMLFormatter(FormatString.__add__(FormatString(value), self).as_str())
|
|
137
|
+
|
|
138
|
+
def as_str(self) -> str:
|
|
139
|
+
"""Returns self as a standart string."""
|
|
140
|
+
return self.__str__()
|
|
141
|
+
|
|
142
|
+
def format(self, *args: object, **kwargs: object) -> "HTMLFormatter":
|
|
143
|
+
return self.STRING_FORMATTER.format(self, *args, **kwargs)
|
|
144
|
+
|
|
145
|
+
|
|
146
|
+
class EscapedString(FormatString):
|
|
147
|
+
@property
|
|
148
|
+
def former_string(self) -> str:
|
|
149
|
+
return html.unescape(self)
|
|
150
|
+
|
|
151
|
+
|
|
152
|
+
class TagFormat(FormatString):
|
|
153
|
+
tag: str
|
|
154
|
+
data: dict[str, typing.Any]
|
|
155
|
+
|
|
156
|
+
def __new__(
|
|
157
|
+
cls,
|
|
158
|
+
string: str,
|
|
159
|
+
/,
|
|
160
|
+
*,
|
|
161
|
+
tag: str,
|
|
162
|
+
**data: typing.Any | None,
|
|
163
|
+
) -> typing.Self:
|
|
164
|
+
if isinstance(string, TagFormat):
|
|
165
|
+
string = string.formatting()
|
|
166
|
+
elif not isinstance(
|
|
167
|
+
string,
|
|
168
|
+
(
|
|
169
|
+
EscapedString,
|
|
170
|
+
HTMLFormatter,
|
|
171
|
+
),
|
|
172
|
+
):
|
|
173
|
+
string = escape(string)
|
|
174
|
+
|
|
175
|
+
obj = super().__new__(cls, string)
|
|
176
|
+
obj.tag = tag
|
|
177
|
+
obj.data = data
|
|
178
|
+
return obj
|
|
179
|
+
|
|
180
|
+
def get_tag_data(self) -> str:
|
|
181
|
+
return "".join(f" {k}={v}" if v is not None else f" {k}" for k, v in self.data.items())
|
|
182
|
+
|
|
183
|
+
def formatting(self) -> HTMLFormatter:
|
|
184
|
+
return HTMLFormatter(
|
|
185
|
+
TAG_FORMAT.format(
|
|
186
|
+
tag=self.tag,
|
|
187
|
+
data=self.get_tag_data(),
|
|
188
|
+
content=self,
|
|
189
|
+
)
|
|
190
|
+
)
|
|
191
|
+
|
|
192
|
+
|
|
193
|
+
class HTMLFormatter(FormatString):
|
|
194
|
+
""">>> HTMLFormatter(bold("Hello, World"))
|
|
195
|
+
>>> '<b>Hello, World</b>'
|
|
196
|
+
>>> HTMLFormatter("Hi, {name:italic}").format(name="Max")
|
|
197
|
+
>>> 'Hi, <i>Max</i>'
|
|
198
|
+
"""
|
|
199
|
+
|
|
200
|
+
PARSE_MODE = ParseMode.HTML
|
|
201
|
+
|
|
202
|
+
|
|
203
|
+
def escape(string: str) -> EscapedString:
|
|
204
|
+
if isinstance(string, EscapedString | HTMLFormatter):
|
|
205
|
+
return EscapedString(string)
|
|
206
|
+
return EscapedString(html.escape(string, quote=False))
|
|
207
|
+
|
|
208
|
+
|
|
209
|
+
def block_quote(string: str, expandable: bool = False) -> TagFormat:
|
|
210
|
+
return TagFormat(
|
|
211
|
+
string,
|
|
212
|
+
tag="blockquote",
|
|
213
|
+
**{} if not expandable else {"expandable": None},
|
|
214
|
+
)
|
|
215
|
+
|
|
216
|
+
|
|
217
|
+
def bold(string: str) -> TagFormat:
|
|
218
|
+
return TagFormat(string, tag="b")
|
|
219
|
+
|
|
220
|
+
|
|
221
|
+
def code_inline(string: str) -> TagFormat:
|
|
222
|
+
return TagFormat(string, tag="code")
|
|
223
|
+
|
|
224
|
+
|
|
225
|
+
def italic(string: str) -> TagFormat:
|
|
226
|
+
return TagFormat(string, tag="i")
|
|
227
|
+
|
|
228
|
+
|
|
229
|
+
def link(href: str, string: str | None = None) -> TagFormat:
|
|
230
|
+
return TagFormat(
|
|
231
|
+
string or href,
|
|
232
|
+
tag="a",
|
|
233
|
+
href=QUOT_MARK + href + QUOT_MARK,
|
|
234
|
+
)
|
|
235
|
+
|
|
236
|
+
|
|
237
|
+
def pre_code(string: str, lang: str | ProgrammingLanguage | None = None) -> TagFormat:
|
|
238
|
+
if lang is None:
|
|
239
|
+
return TagFormat(string, tag="pre")
|
|
240
|
+
lang = lang.value if isinstance(lang, ProgrammingLanguage) else lang
|
|
241
|
+
return pre_code(TagFormat(string, tag="code", **{"class": f"language-{lang}"}))
|
|
242
|
+
|
|
243
|
+
|
|
244
|
+
def spoiler(string: str) -> TagFormat:
|
|
245
|
+
return TagFormat(string, tag="tg-spoiler")
|
|
246
|
+
|
|
247
|
+
|
|
248
|
+
def strike(string: str) -> TagFormat:
|
|
249
|
+
return TagFormat(string, tag="s")
|
|
250
|
+
|
|
251
|
+
|
|
252
|
+
def mention(string: str, user_id: int) -> TagFormat:
|
|
253
|
+
return link(tg_mention_link(user_id=user_id), string)
|
|
254
|
+
|
|
255
|
+
|
|
256
|
+
def tg_emoji(string: str, emoji_id: int) -> TagFormat:
|
|
257
|
+
return TagFormat(string, tag="tg-emoji", emoji_id=emoji_id)
|
|
258
|
+
|
|
259
|
+
|
|
260
|
+
def underline(string: str) -> TagFormat:
|
|
261
|
+
return TagFormat(string, tag="u")
|
|
262
|
+
|
|
263
|
+
|
|
264
|
+
__all__ = (
|
|
282
265
|
"FormatString",
|
|
283
266
|
"HTMLFormatter",
|
|
284
267
|
"SpecialFormat",
|
|
285
268
|
"block_quote",
|
|
286
269
|
"bold",
|
|
287
|
-
"channel_boost_link",
|
|
288
270
|
"code_inline",
|
|
289
271
|
"escape",
|
|
290
|
-
"get_channel_boost_link",
|
|
291
|
-
"get_invite_chat_link",
|
|
292
|
-
"get_mention_link",
|
|
293
|
-
"get_resolve_domain_link",
|
|
294
|
-
"get_start_bot_link",
|
|
295
|
-
"get_start_group_link",
|
|
296
|
-
"invite_chat_link",
|
|
297
272
|
"italic",
|
|
298
273
|
"link",
|
|
299
274
|
"mention",
|
|
300
275
|
"pre_code",
|
|
301
|
-
"resolve_domain",
|
|
302
276
|
"spoiler",
|
|
303
|
-
"start_bot_link",
|
|
304
|
-
"start_group_link",
|
|
305
277
|
"strike",
|
|
306
278
|
"tg_emoji",
|
|
307
|
-
"underline",
|
|
308
|
-
)
|
|
279
|
+
"underline",
|
|
280
|
+
)
|