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,284 +1,284 @@
1
- import html
2
- import string
3
- import typing
4
- from contextlib import suppress
5
-
6
- from telegrinder.tools.parse_mode import ParseMode
7
- from telegrinder.types.enums import ProgrammingLanguage
8
-
9
- from .links import (
10
- get_channel_boost_link,
11
- get_invite_chat_link,
12
- get_mention_link,
13
- get_resolve_domain_link,
14
- get_start_bot_link,
15
- get_start_group_link,
16
- )
17
- from .spec_html_formats import SpecialFormat, is_spec_format
18
-
19
- TAG_FORMAT = "<{tag}{data}>{content}</{tag}>"
20
- QUOT_MARK = '"'
21
-
22
-
23
- class StringFormatterProto(typing.Protocol):
24
- def format_field(self, value: typing.Any, fmt: str) -> "HTMLFormatter": ...
25
-
26
- def format(self, __string: str, *args: object, **kwargs: object) -> "HTMLFormatter": ...
27
-
28
-
29
- class StringFormatter(string.Formatter):
30
- """String formatter, using substitutions from args and kwargs.
31
- The substitutions are identified by braces ('{' and '}') with
32
- specifiers: `bold`, `italic`, etc.
33
- """
34
-
35
- __formats__: typing.ClassVar[tuple[str, ...]] = (
36
- "blockquote",
37
- "bold",
38
- "code_inline",
39
- "italic",
40
- "spoiler",
41
- "strike",
42
- "underline",
43
- )
44
-
45
- def is_html_format(self, value: typing.Any, fmt: str) -> str:
46
- if not fmt:
47
- raise ValueError("Formats union should be: format+format.")
48
- if fmt not in self.__formats__:
49
- raise ValueError(
50
- "Unknown format {!r} for object of type {!r}.".format(
51
- fmt,
52
- type(value).__name__,
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 check_formats(self, value: typing.Any, fmts: list[str]) -> "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)(**value.__dict__).formatting()
91
- if is_spec_format(value)
92
- else value
93
- )
94
- ),
95
- fmt,
96
- )
97
- )
98
- return self.format_raw_value(value, fmt)
99
-
100
- def format_raw_value(self, value: typing.Any, fmt: str) -> "HTMLFormatter":
101
- fmts = list(map(lambda fmt: self.is_html_format(value, fmt), fmt.split("+")))
102
- tag_format = self.check_formats(value, fmts)
103
-
104
- if is_spec_format(value):
105
- value.string = tag_format
106
- tag_format = self.get_spec_formatter(value)(**value.__dict__)
107
-
108
- return tag_format.formatting()
109
-
110
- def format(self, __string: str, *args: object, **kwargs: object) -> "HTMLFormatter":
111
- return HTMLFormatter(super().format(__string, *args, **kwargs))
112
-
113
-
114
- class FormatString(str):
115
- STRING_FORMATTER: StringFormatterProto = StringFormatter()
116
-
117
- def __new__(cls, string: str) -> typing.Self:
118
- if isinstance(string, TagFormat):
119
- return super().__new__(cls, string.formatting())
120
- return super().__new__(cls, string)
121
-
122
- def __add__(self, value: str) -> "HTMLFormatter":
123
- return HTMLFormatter(
124
- str.__add__(
125
- escape(self),
126
- value.formatting() if isinstance(value, TagFormat) else escape(value),
127
- )
128
- )
129
-
130
- def __radd__(self, value: str) -> "HTMLFormatter":
131
- """Return value+self."""
132
- return HTMLFormatter(FormatString.__add__(FormatString(value), self).as_str())
133
-
134
- def as_str(self) -> str:
135
- """Return self as a standart string."""
136
- return self.__str__()
137
-
138
- def format(self, *args: object, **kwargs: object) -> "HTMLFormatter":
139
- return self.STRING_FORMATTER.format(self, *args, **kwargs)
140
-
141
-
142
- class EscapedString(FormatString):
143
- @property
144
- def former_string(self) -> str:
145
- return html.unescape(self)
146
-
147
-
148
- class TagFormat(FormatString):
149
- tag: str
150
- data: dict[str, typing.Any]
151
-
152
- def __new__(
153
- cls,
154
- string: str,
155
- *,
156
- tag: str,
157
- **data: typing.Any,
158
- ) -> typing.Self:
159
- if isinstance(string, TagFormat):
160
- string = string.formatting()
161
- elif not isinstance(
162
- string,
163
- (
164
- EscapedString,
165
- HTMLFormatter,
166
- ),
167
- ):
168
- string = escape(string)
169
- obj = super().__new__(cls, string)
170
- obj.tag = tag
171
- obj.data = data
172
- return obj
173
-
174
- def get_tag_data(self) -> str:
175
- return "".join(f" {k}={v}" for k, v in self.data.items())
176
-
177
- def formatting(self) -> "HTMLFormatter":
178
- return HTMLFormatter(
179
- TAG_FORMAT.format(
180
- tag=self.tag,
181
- data=self.get_tag_data(),
182
- content=self,
183
- )
184
- )
185
-
186
-
187
- class HTMLFormatter(FormatString):
188
- """
189
- >>> HTMLFormatter(bold("Hello, World"))
190
- '<b>Hello, World</b>'
191
- HTMLFormatter("Hi, {name:italic}").format(name="Max")
192
- 'Hi, <i>Max</i>'
193
- """
194
-
195
- PARSE_MODE = ParseMode.HTML
196
-
197
-
198
- def escape(string: str) -> EscapedString:
199
- if isinstance(string, EscapedString | HTMLFormatter):
200
- return EscapedString(string)
201
- return EscapedString(html.escape(string, quote=False))
202
-
203
-
204
- def block_quote(string: str) -> TagFormat:
205
- return TagFormat(string, tag="blockquote")
206
-
207
-
208
- def bold(string: str) -> TagFormat:
209
- return TagFormat(string, tag="b")
210
-
211
-
212
- def channel_boost_link(channel_id: str | int, string: str | None = None):
213
- return link(get_channel_boost_link(channel_id), string)
214
-
215
-
216
- def code_inline(string: str) -> TagFormat:
217
- return TagFormat(string, tag="code")
218
-
219
-
220
- def italic(string: str) -> TagFormat:
221
- return TagFormat(string, tag="i")
222
-
223
-
224
- def link(href: str, string: str | None = None) -> TagFormat:
225
- return TagFormat(
226
- string or href,
227
- tag="a",
228
- href=QUOT_MARK + href + QUOT_MARK,
229
- )
230
-
231
-
232
- def pre_code(string: str, lang: str | ProgrammingLanguage | None = None) -> TagFormat:
233
- if lang is None:
234
- return TagFormat(string, tag="pre")
235
- lang = lang.value if isinstance(lang, ProgrammingLanguage) else lang
236
- return pre_code(TagFormat(string, tag="code", **{"class": f"language-{lang}"}))
237
-
238
-
239
- def spoiler(string: str) -> TagFormat:
240
- return TagFormat(string, tag="tg-spoiler")
241
-
242
-
243
- def start_bot_link(bot_id: str | int, data: str, string: str | None = None) -> TagFormat:
244
- return link(get_start_bot_link(bot_id, data), string)
245
-
246
-
247
- def start_group_link(bot_id: str | int, data: str, string: str | None = None) -> TagFormat:
248
- return link(get_start_group_link(bot_id, data), string)
249
-
250
-
251
- def strike(string: str) -> TagFormat:
252
- return TagFormat(string, tag="s")
253
-
254
-
255
- def mention(string: str, user_id: int) -> TagFormat:
256
- return link(get_mention_link(user_id), string)
257
-
258
-
259
- def tg_emoji(string: str, emoji_id: int) -> TagFormat:
260
- return TagFormat(string, tag="tg-emoji", **{"emoji-id": emoji_id})
261
-
262
-
263
- def invite_chat_link(invite_link: str, string: str | None = None) -> TagFormat:
264
- return link(
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
+ import html
2
+ import string
3
+ import typing
4
+ from contextlib import suppress
5
+
6
+ from telegrinder.tools.parse_mode import ParseMode
7
+ from telegrinder.types.enums import ProgrammingLanguage
8
+
9
+ from .links import (
10
+ get_channel_boost_link,
11
+ get_invite_chat_link,
12
+ get_mention_link,
13
+ get_resolve_domain_link,
14
+ get_start_bot_link,
15
+ get_start_group_link,
16
+ )
17
+ from .spec_html_formats import SpecialFormat, is_spec_format
18
+
19
+ TAG_FORMAT = "<{tag}{data}>{content}</{tag}>"
20
+ QUOT_MARK = '"'
21
+
22
+
23
+ class StringFormatterProto(typing.Protocol):
24
+ def format_field(self, value: typing.Any, fmt: str) -> "HTMLFormatter": ...
25
+
26
+ def format(self, __string: str, *args: object, **kwargs: object) -> "HTMLFormatter": ...
27
+
28
+
29
+ class StringFormatter(string.Formatter):
30
+ """String formatter, using substitutions from args and kwargs.
31
+ The substitutions are identified by braces ('{' and '}') with
32
+ specifiers: `bold`, `italic`, etc.
33
+ """
34
+
35
+ __formats__: typing.ClassVar[tuple[str, ...]] = (
36
+ "blockquote",
37
+ "bold",
38
+ "code_inline",
39
+ "italic",
40
+ "spoiler",
41
+ "strike",
42
+ "underline",
43
+ )
44
+
45
+ def is_html_format(self, value: typing.Any, fmt: str) -> str:
46
+ if not fmt:
47
+ raise ValueError("Formats union should be: format+format.")
48
+ if fmt not in self.__formats__:
49
+ raise ValueError(
50
+ "Unknown format {!r} for object of type {!r}.".format(
51
+ fmt,
52
+ type(value).__name__,
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 check_formats(self, value: typing.Any, fmts: list[str]) -> "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)(**value.__dict__).formatting()
91
+ if is_spec_format(value)
92
+ else value
93
+ )
94
+ ),
95
+ fmt,
96
+ )
97
+ )
98
+ return self.format_raw_value(value, fmt)
99
+
100
+ def format_raw_value(self, value: typing.Any, fmt: str) -> "HTMLFormatter":
101
+ fmts = list(map(lambda fmt: self.is_html_format(value, fmt), fmt.split("+")))
102
+ tag_format = self.check_formats(value, fmts)
103
+
104
+ if is_spec_format(value):
105
+ value.string = tag_format
106
+ tag_format = self.get_spec_formatter(value)(**value.__dict__)
107
+
108
+ return tag_format.formatting()
109
+
110
+ def format(self, __string: str, *args: object, **kwargs: object) -> "HTMLFormatter":
111
+ return HTMLFormatter(super().format(__string, *args, **kwargs))
112
+
113
+
114
+ class FormatString(str):
115
+ STRING_FORMATTER: StringFormatterProto = StringFormatter()
116
+
117
+ def __new__(cls, string: str) -> typing.Self:
118
+ if isinstance(string, TagFormat):
119
+ return super().__new__(cls, string.formatting())
120
+ return super().__new__(cls, string)
121
+
122
+ def __add__(self, value: str) -> "HTMLFormatter":
123
+ return HTMLFormatter(
124
+ str.__add__(
125
+ escape(self),
126
+ value.formatting() if isinstance(value, TagFormat) else escape(value),
127
+ )
128
+ )
129
+
130
+ def __radd__(self, value: str) -> "HTMLFormatter":
131
+ """Return value+self."""
132
+ return HTMLFormatter(FormatString.__add__(FormatString(value), self).as_str())
133
+
134
+ def as_str(self) -> str:
135
+ """Return self as a standart string."""
136
+ return self.__str__()
137
+
138
+ def format(self, *args: object, **kwargs: object) -> "HTMLFormatter":
139
+ return self.STRING_FORMATTER.format(self, *args, **kwargs)
140
+
141
+
142
+ class EscapedString(FormatString):
143
+ @property
144
+ def former_string(self) -> str:
145
+ return html.unescape(self)
146
+
147
+
148
+ class TagFormat(FormatString):
149
+ tag: str
150
+ data: dict[str, typing.Any]
151
+
152
+ def __new__(
153
+ cls,
154
+ string: str,
155
+ *,
156
+ tag: str,
157
+ **data: typing.Any,
158
+ ) -> typing.Self:
159
+ if isinstance(string, TagFormat):
160
+ string = string.formatting()
161
+ elif not isinstance(
162
+ string,
163
+ (
164
+ EscapedString,
165
+ HTMLFormatter,
166
+ ),
167
+ ):
168
+ string = escape(string)
169
+ obj = super().__new__(cls, string)
170
+ obj.tag = tag
171
+ obj.data = data
172
+ return obj
173
+
174
+ def get_tag_data(self) -> str:
175
+ return "".join(f" {k}={v}" for k, v in self.data.items())
176
+
177
+ def formatting(self) -> "HTMLFormatter":
178
+ return HTMLFormatter(
179
+ TAG_FORMAT.format(
180
+ tag=self.tag,
181
+ data=self.get_tag_data(),
182
+ content=self,
183
+ )
184
+ )
185
+
186
+
187
+ class HTMLFormatter(FormatString):
188
+ """
189
+ >>> HTMLFormatter(bold("Hello, World"))
190
+ '<b>Hello, World</b>'
191
+ HTMLFormatter("Hi, {name:italic}").format(name="Max")
192
+ 'Hi, <i>Max</i>'
193
+ """
194
+
195
+ PARSE_MODE = ParseMode.HTML
196
+
197
+
198
+ def escape(string: str) -> EscapedString:
199
+ if isinstance(string, EscapedString | HTMLFormatter):
200
+ return EscapedString(string)
201
+ return EscapedString(html.escape(string, quote=False))
202
+
203
+
204
+ def block_quote(string: str) -> TagFormat:
205
+ return TagFormat(string, tag="blockquote")
206
+
207
+
208
+ def bold(string: str) -> TagFormat:
209
+ return TagFormat(string, tag="b")
210
+
211
+
212
+ def channel_boost_link(channel_id: str | int, string: str | None = None):
213
+ return link(get_channel_boost_link(channel_id), string)
214
+
215
+
216
+ def code_inline(string: str) -> TagFormat:
217
+ return TagFormat(string, tag="code")
218
+
219
+
220
+ def italic(string: str) -> TagFormat:
221
+ return TagFormat(string, tag="i")
222
+
223
+
224
+ def link(href: str, string: str | None = None) -> TagFormat:
225
+ return TagFormat(
226
+ string or href,
227
+ tag="a",
228
+ href=QUOT_MARK + href + QUOT_MARK,
229
+ )
230
+
231
+
232
+ def pre_code(string: str, lang: str | ProgrammingLanguage | None = None) -> TagFormat:
233
+ if lang is None:
234
+ return TagFormat(string, tag="pre")
235
+ lang = lang.value if isinstance(lang, ProgrammingLanguage) else lang
236
+ return pre_code(TagFormat(string, tag="code", **{"class": f"language-{lang}"}))
237
+
238
+
239
+ def spoiler(string: str) -> TagFormat:
240
+ return TagFormat(string, tag="tg-spoiler")
241
+
242
+
243
+ def start_bot_link(bot_id: str | int, data: str, string: str | None = None) -> TagFormat:
244
+ return link(get_start_bot_link(bot_id, data), string)
245
+
246
+
247
+ def start_group_link(bot_id: str | int, data: str, string: str | None = None) -> TagFormat:
248
+ return link(get_start_group_link(bot_id, data), string)
249
+
250
+
251
+ def strike(string: str) -> TagFormat:
252
+ return TagFormat(string, tag="s")
253
+
254
+
255
+ def mention(string: str, user_id: int) -> TagFormat:
256
+ return link(get_mention_link(user_id), string)
257
+
258
+
259
+ def tg_emoji(string: str, emoji_id: int) -> TagFormat:
260
+ return TagFormat(string, tag="tg-emoji", **{"emoji-id": emoji_id})
261
+
262
+
263
+ def invite_chat_link(invite_link: str, string: str | None = None) -> TagFormat:
264
+ return link(
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__ = (
282
282
  "FormatString",
283
283
  "HTMLFormatter",
284
284
  "SpecialFormat",
@@ -304,5 +304,5 @@ __all__ = (
304
304
  "start_group_link",
305
305
  "strike",
306
306
  "tg_emoji",
307
- "underline",
308
- )
307
+ "underline",
308
+ )
@@ -1,38 +1,38 @@
1
- def get_mention_link(user_id: int) -> str:
2
- return f"tg://user?id={user_id}"
3
-
4
-
5
- def get_resolve_domain_link(username: str) -> str:
6
- return f"tg://resolve?domain={username}"
7
-
8
-
9
- def get_start_bot_link(bot_id: str | int, data: str) -> str:
10
- if isinstance(bot_id, int):
11
- return get_mention_link(bot_id) + f"&start={data}"
12
- return get_resolve_domain_link(bot_id) + f"&start={data}"
13
-
14
-
15
- def get_start_group_link(bot_id: str | int, data: str) -> str:
16
- if isinstance(bot_id, int):
17
- return get_mention_link(bot_id) + f"&startgroup={data}"
18
- return get_resolve_domain_link(bot_id) + f"&startgroup={data}"
19
-
20
-
21
- def get_channel_boost_link(channel_id: str | int) -> str:
22
- if isinstance(channel_id, int):
23
- return get_mention_link(channel_id) + "&boost"
24
- return get_resolve_domain_link(channel_id) + "&boost"
25
-
26
-
27
- def get_invite_chat_link(invite_link: str) -> str:
28
- return f"tg://join?invite={invite_link}"
29
-
30
-
31
- __all__ = (
1
+ def get_mention_link(user_id: int) -> str:
2
+ return f"tg://user?id={user_id}"
3
+
4
+
5
+ def get_resolve_domain_link(username: str) -> str:
6
+ return f"tg://resolve?domain={username}"
7
+
8
+
9
+ def get_start_bot_link(bot_id: str | int, data: str) -> str:
10
+ if isinstance(bot_id, int):
11
+ return get_mention_link(bot_id) + f"&start={data}"
12
+ return get_resolve_domain_link(bot_id) + f"&start={data}"
13
+
14
+
15
+ def get_start_group_link(bot_id: str | int, data: str) -> str:
16
+ if isinstance(bot_id, int):
17
+ return get_mention_link(bot_id) + f"&startgroup={data}"
18
+ return get_resolve_domain_link(bot_id) + f"&startgroup={data}"
19
+
20
+
21
+ def get_channel_boost_link(channel_id: str | int) -> str:
22
+ if isinstance(channel_id, int):
23
+ return get_mention_link(channel_id) + "&boost"
24
+ return get_resolve_domain_link(channel_id) + "&boost"
25
+
26
+
27
+ def get_invite_chat_link(invite_link: str) -> str:
28
+ return f"tg://join?invite={invite_link}"
29
+
30
+
31
+ __all__ = (
32
32
  "get_channel_boost_link",
33
33
  "get_invite_chat_link",
34
34
  "get_mention_link",
35
35
  "get_resolve_domain_link",
36
36
  "get_start_bot_link",
37
- "get_start_group_link",
38
- )
37
+ "get_start_group_link",
38
+ )