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.

Files changed (192) hide show
  1. telegrinder/__init__.py +148 -149
  2. telegrinder/api/__init__.py +9 -8
  3. telegrinder/api/api.py +101 -93
  4. telegrinder/api/error.py +20 -16
  5. telegrinder/api/response.py +20 -20
  6. telegrinder/api/token.py +36 -36
  7. telegrinder/bot/__init__.py +72 -66
  8. telegrinder/bot/bot.py +83 -76
  9. telegrinder/bot/cute_types/__init__.py +19 -17
  10. telegrinder/bot/cute_types/base.py +184 -258
  11. telegrinder/bot/cute_types/callback_query.py +400 -385
  12. telegrinder/bot/cute_types/chat_join_request.py +62 -61
  13. telegrinder/bot/cute_types/chat_member_updated.py +157 -160
  14. telegrinder/bot/cute_types/inline_query.py +44 -43
  15. telegrinder/bot/cute_types/message.py +2590 -2637
  16. telegrinder/bot/cute_types/pre_checkout_query.py +42 -0
  17. telegrinder/bot/cute_types/update.py +112 -104
  18. telegrinder/bot/cute_types/utils.py +62 -95
  19. telegrinder/bot/dispatch/__init__.py +59 -55
  20. telegrinder/bot/dispatch/abc.py +76 -77
  21. telegrinder/bot/dispatch/context.py +96 -98
  22. telegrinder/bot/dispatch/dispatch.py +254 -202
  23. telegrinder/bot/dispatch/handler/__init__.py +13 -13
  24. telegrinder/bot/dispatch/handler/abc.py +23 -24
  25. telegrinder/bot/dispatch/handler/audio_reply.py +44 -44
  26. telegrinder/bot/dispatch/handler/base.py +57 -57
  27. telegrinder/bot/dispatch/handler/document_reply.py +44 -44
  28. telegrinder/bot/dispatch/handler/func.py +129 -135
  29. telegrinder/bot/dispatch/handler/media_group_reply.py +44 -43
  30. telegrinder/bot/dispatch/handler/message_reply.py +36 -36
  31. telegrinder/bot/dispatch/handler/photo_reply.py +44 -44
  32. telegrinder/bot/dispatch/handler/sticker_reply.py +37 -37
  33. telegrinder/bot/dispatch/handler/video_reply.py +44 -44
  34. telegrinder/bot/dispatch/middleware/__init__.py +3 -3
  35. telegrinder/bot/dispatch/middleware/abc.py +97 -22
  36. telegrinder/bot/dispatch/middleware/global_middleware.py +70 -0
  37. telegrinder/bot/dispatch/process.py +151 -157
  38. telegrinder/bot/dispatch/return_manager/__init__.py +15 -13
  39. telegrinder/bot/dispatch/return_manager/abc.py +104 -108
  40. telegrinder/bot/dispatch/return_manager/callback_query.py +20 -20
  41. telegrinder/bot/dispatch/return_manager/inline_query.py +15 -15
  42. telegrinder/bot/dispatch/return_manager/message.py +36 -36
  43. telegrinder/bot/dispatch/return_manager/pre_checkout_query.py +20 -0
  44. telegrinder/bot/dispatch/view/__init__.py +15 -13
  45. telegrinder/bot/dispatch/view/abc.py +45 -41
  46. telegrinder/bot/dispatch/view/base.py +231 -200
  47. telegrinder/bot/dispatch/view/box.py +140 -129
  48. telegrinder/bot/dispatch/view/callback_query.py +16 -17
  49. telegrinder/bot/dispatch/view/chat_join_request.py +11 -16
  50. telegrinder/bot/dispatch/view/chat_member.py +37 -39
  51. telegrinder/bot/dispatch/view/inline_query.py +16 -17
  52. telegrinder/bot/dispatch/view/message.py +43 -44
  53. telegrinder/bot/dispatch/view/pre_checkout_query.py +16 -0
  54. telegrinder/bot/dispatch/view/raw.py +116 -114
  55. telegrinder/bot/dispatch/waiter_machine/__init__.py +17 -17
  56. telegrinder/bot/dispatch/waiter_machine/actions.py +14 -13
  57. telegrinder/bot/dispatch/waiter_machine/hasher/__init__.py +8 -8
  58. telegrinder/bot/dispatch/waiter_machine/hasher/callback.py +55 -55
  59. telegrinder/bot/dispatch/waiter_machine/hasher/hasher.py +59 -57
  60. telegrinder/bot/dispatch/waiter_machine/hasher/message.py +51 -51
  61. telegrinder/bot/dispatch/waiter_machine/hasher/state.py +20 -19
  62. telegrinder/bot/dispatch/waiter_machine/machine.py +251 -172
  63. telegrinder/bot/dispatch/waiter_machine/middleware.py +94 -89
  64. telegrinder/bot/dispatch/waiter_machine/short_state.py +57 -68
  65. telegrinder/bot/polling/__init__.py +4 -4
  66. telegrinder/bot/polling/abc.py +25 -25
  67. telegrinder/bot/polling/polling.py +139 -131
  68. telegrinder/bot/rules/__init__.py +85 -62
  69. telegrinder/bot/rules/abc.py +213 -206
  70. telegrinder/bot/rules/callback_data.py +122 -163
  71. telegrinder/bot/rules/chat_join.py +45 -43
  72. telegrinder/bot/rules/command.py +126 -126
  73. telegrinder/bot/rules/enum_text.py +33 -36
  74. telegrinder/bot/rules/func.py +28 -26
  75. telegrinder/bot/rules/fuzzy.py +24 -24
  76. telegrinder/bot/rules/id.py +24 -0
  77. telegrinder/bot/rules/inline.py +58 -56
  78. telegrinder/bot/rules/integer.py +21 -20
  79. telegrinder/bot/rules/is_from.py +127 -127
  80. telegrinder/bot/rules/logic.py +18 -0
  81. telegrinder/bot/rules/markup.py +42 -43
  82. telegrinder/bot/rules/mention.py +14 -14
  83. telegrinder/bot/rules/message.py +15 -17
  84. telegrinder/bot/rules/message_entities.py +33 -35
  85. telegrinder/bot/rules/node.py +33 -27
  86. telegrinder/bot/rules/payload.py +81 -0
  87. telegrinder/bot/rules/payment_invoice.py +29 -0
  88. telegrinder/bot/rules/regex.py +36 -37
  89. telegrinder/bot/rules/rule_enum.py +72 -72
  90. telegrinder/bot/rules/start.py +42 -42
  91. telegrinder/bot/rules/state.py +35 -37
  92. telegrinder/bot/rules/text.py +38 -33
  93. telegrinder/bot/rules/update.py +15 -15
  94. telegrinder/bot/scenario/__init__.py +5 -5
  95. telegrinder/bot/scenario/abc.py +17 -19
  96. telegrinder/bot/scenario/checkbox.py +174 -176
  97. telegrinder/bot/scenario/choice.py +48 -51
  98. telegrinder/client/__init__.py +12 -4
  99. telegrinder/client/abc.py +100 -75
  100. telegrinder/client/aiohttp.py +134 -130
  101. telegrinder/client/form_data.py +31 -0
  102. telegrinder/client/sonic.py +212 -0
  103. telegrinder/model.py +208 -315
  104. telegrinder/modules.py +239 -237
  105. telegrinder/msgspec_json.py +14 -14
  106. telegrinder/msgspec_utils.py +478 -410
  107. telegrinder/node/__init__.py +86 -25
  108. telegrinder/node/attachment.py +163 -87
  109. telegrinder/node/base.py +288 -160
  110. telegrinder/node/callback_query.py +54 -53
  111. telegrinder/node/command.py +34 -33
  112. telegrinder/node/composer.py +163 -198
  113. telegrinder/node/container.py +33 -27
  114. telegrinder/node/either.py +82 -0
  115. telegrinder/node/event.py +54 -65
  116. telegrinder/node/file.py +51 -0
  117. telegrinder/node/me.py +15 -16
  118. telegrinder/node/payload.py +78 -0
  119. telegrinder/node/polymorphic.py +67 -48
  120. telegrinder/node/rule.py +72 -76
  121. telegrinder/node/scope.py +36 -38
  122. telegrinder/node/source.py +87 -71
  123. telegrinder/node/text.py +53 -41
  124. telegrinder/node/tools/__init__.py +3 -3
  125. telegrinder/node/tools/generator.py +36 -40
  126. telegrinder/py.typed +0 -0
  127. telegrinder/rules.py +1 -62
  128. telegrinder/tools/__init__.py +152 -93
  129. telegrinder/tools/adapter/__init__.py +19 -0
  130. telegrinder/tools/adapter/abc.py +49 -0
  131. telegrinder/tools/adapter/dataclass.py +56 -0
  132. telegrinder/{bot/rules → tools}/adapter/errors.py +5 -5
  133. telegrinder/{bot/rules → tools}/adapter/event.py +63 -65
  134. telegrinder/{bot/rules → tools}/adapter/node.py +46 -48
  135. telegrinder/{bot/rules → tools}/adapter/raw_event.py +27 -27
  136. telegrinder/{bot/rules → tools}/adapter/raw_update.py +30 -30
  137. telegrinder/tools/buttons.py +106 -80
  138. telegrinder/tools/callback_data_serilization/__init__.py +5 -0
  139. telegrinder/tools/callback_data_serilization/abc.py +51 -0
  140. telegrinder/tools/callback_data_serilization/json_ser.py +60 -0
  141. telegrinder/tools/callback_data_serilization/msgpack_ser.py +172 -0
  142. telegrinder/tools/error_handler/__init__.py +7 -7
  143. telegrinder/tools/error_handler/abc.py +30 -33
  144. telegrinder/tools/error_handler/error.py +9 -9
  145. telegrinder/tools/error_handler/error_handler.py +179 -193
  146. telegrinder/tools/formatting/__init__.py +83 -63
  147. telegrinder/tools/formatting/deep_links.py +541 -0
  148. telegrinder/tools/formatting/{html.py → html_formatter.py} +266 -294
  149. telegrinder/tools/formatting/spec_html_formats.py +71 -117
  150. telegrinder/tools/functional.py +8 -12
  151. telegrinder/tools/global_context/__init__.py +7 -7
  152. telegrinder/tools/global_context/abc.py +63 -63
  153. telegrinder/tools/global_context/global_context.py +387 -412
  154. telegrinder/tools/global_context/telegrinder_ctx.py +27 -27
  155. telegrinder/tools/i18n/__init__.py +7 -7
  156. telegrinder/tools/i18n/abc.py +30 -30
  157. telegrinder/tools/i18n/middleware/__init__.py +3 -3
  158. telegrinder/tools/i18n/middleware/abc.py +22 -25
  159. telegrinder/tools/i18n/simple.py +43 -43
  160. telegrinder/tools/input_file_directory.py +30 -0
  161. telegrinder/tools/keyboard.py +128 -128
  162. telegrinder/tools/lifespan.py +105 -0
  163. telegrinder/tools/limited_dict.py +32 -37
  164. telegrinder/tools/loop_wrapper/__init__.py +4 -4
  165. telegrinder/tools/loop_wrapper/abc.py +20 -15
  166. telegrinder/tools/loop_wrapper/loop_wrapper.py +169 -224
  167. telegrinder/tools/magic.py +307 -157
  168. telegrinder/tools/parse_mode.py +6 -6
  169. telegrinder/tools/state_storage/__init__.py +4 -4
  170. telegrinder/tools/state_storage/abc.py +31 -35
  171. telegrinder/tools/state_storage/memory.py +25 -25
  172. telegrinder/tools/strings.py +13 -0
  173. telegrinder/types/__init__.py +268 -260
  174. telegrinder/types/enums.py +711 -701
  175. telegrinder/types/input_file.py +51 -0
  176. telegrinder/types/methods.py +5055 -4633
  177. telegrinder/types/objects.py +7058 -6950
  178. telegrinder/verification_utils.py +30 -32
  179. {telegrinder-0.3.4.dist-info → telegrinder-0.4.0.dist-info}/LICENSE +22 -22
  180. telegrinder-0.4.0.dist-info/METADATA +144 -0
  181. telegrinder-0.4.0.dist-info/RECORD +182 -0
  182. {telegrinder-0.3.4.dist-info → telegrinder-0.4.0.dist-info}/WHEEL +1 -1
  183. telegrinder/bot/rules/adapter/__init__.py +0 -17
  184. telegrinder/bot/rules/adapter/abc.py +0 -31
  185. telegrinder/node/message.py +0 -14
  186. telegrinder/node/update.py +0 -15
  187. telegrinder/tools/formatting/links.py +0 -38
  188. telegrinder/tools/kb_set/__init__.py +0 -4
  189. telegrinder/tools/kb_set/base.py +0 -15
  190. telegrinder/tools/kb_set/yaml.py +0 -63
  191. telegrinder-0.3.4.dist-info/METADATA +0 -110
  192. telegrinder-0.3.4.dist-info/RECORD +0 -165
telegrinder/modules.py CHANGED
@@ -1,237 +1,239 @@
1
- import os
2
- import typing
3
-
4
- from choicelib import choice_in_order
5
-
6
-
7
- @typing.runtime_checkable
8
- class LoggerModule(typing.Protocol):
9
- def debug(self, __msg: object, *args: object, **kwargs: object) -> None: ...
10
-
11
- def info(self, __msg: object, *args: object, **kwargs: object) -> None: ...
12
-
13
- def warning(self, __msg: object, *args: object, **kwargs: object) -> None: ...
14
-
15
- def error(self, __msg: object, *args: object, **kwargs: object) -> None: ...
16
-
17
- def critical(self, __msg: object, *args: object, **kwargs: object) -> None: ...
18
-
19
- def exception(self, __msg: object, *args: object, **kwargs: object) -> None: ...
20
-
21
- if typing.TYPE_CHECKING:
22
-
23
- def set_level(
24
- self,
25
- level: typing.Literal[
26
- "DEBUG",
27
- "INFO",
28
- "WARNING",
29
- "ERROR",
30
- "CRITICAL",
31
- "EXCEPTION",
32
- ],
33
- /,
34
- ) -> None: ...
35
-
36
-
37
- logger: LoggerModule
38
- logging_level = os.getenv("LOGGER_LEVEL", default="DEBUG").upper()
39
- logging_module = choice_in_order(["loguru"], default="logging", do_import=False)
40
- asyncio_module = choice_in_order(["uvloop"], default="asyncio", do_import=False)
41
-
42
- if logging_module == "loguru":
43
- import os
44
- import sys
45
-
46
- from loguru import logger # type: ignore
47
-
48
- os.environ.setdefault("LOGURU_AUTOINIT", "0")
49
- log_format = (
50
- "<level>{level: <8}</level> | "
51
- "<lg>{time:YYYY-MM-DD HH:mm:ss}</lg> | "
52
- "<le>{name}</le>:<le>{function}</le>:"
53
- "<le>{line}</le> > <lw>{message}</lw>"
54
- )
55
- logger.remove() # type: ignore
56
- handler_id = logger.add( # type: ignore
57
- sink=sys.stderr,
58
- format=log_format,
59
- enqueue=True,
60
- colorize=True,
61
- level=logging_level,
62
- )
63
-
64
- elif logging_module == "logging":
65
- """
66
- This is a workaround for lazy formatting with {} in logging.
67
- About:
68
- https://docs.python.org/3/howto/logging-cookbook.html#use-of-alternative-formatting-styles
69
- """
70
-
71
- import inspect
72
- import logging
73
- import sys
74
-
75
- import colorama
76
-
77
- colorama.just_fix_windows_console() # init & fix console
78
-
79
- FORMAT = (
80
- "<white>{name: <4} |</white> <level>{levelname: <8}</level>"
81
- " <white>|</white> <green>{asctime}</green> <white>|</white> <level_module>"
82
- "{module}</level_module><white>:</white><level_func>"
83
- "{funcName}</level_func><white>:</white><level_lineno>"
84
- "{lineno}</level_lineno><white> > </white><level_message>"
85
- "{message}</level_message>"
86
- )
87
- COLORS = {
88
- "red": colorama.Fore.LIGHTRED_EX,
89
- "green": colorama.Fore.LIGHTGREEN_EX,
90
- "blue": colorama.Fore.LIGHTBLUE_EX,
91
- "white": colorama.Fore.LIGHTWHITE_EX,
92
- "yellow": colorama.Fore.LIGHTYELLOW_EX,
93
- "magenta": colorama.Fore.LIGHTMAGENTA_EX,
94
- "cyan": colorama.Fore.LIGHTCYAN_EX,
95
- "reset": colorama.Style.RESET_ALL,
96
- }
97
- LEVEL_SETTINGS = {
98
- "INFO": {
99
- "level": "green",
100
- "level_module": "blue",
101
- "level_func": "cyan",
102
- "level_lineno": "white",
103
- "level_message": "green",
104
- },
105
- "DEBUG": {
106
- "level": "blue",
107
- "level_module": "yellow",
108
- "level_func": "green",
109
- "level_lineno": "cyan",
110
- "level_message": "blue",
111
- },
112
- "WARNING": {
113
- "level": "yellow",
114
- "level_module": "red",
115
- "level_func": "green",
116
- "level_lineno": "red",
117
- "level_message": "yellow",
118
- },
119
- "ERROR": {
120
- "level": "red",
121
- "level_module": "magenta",
122
- "level_func": "yellow",
123
- "level_lineno": "green",
124
- "level_message": "red",
125
- },
126
- "CRITICAL": {
127
- "level": "cyan",
128
- "level_module": "yellow",
129
- "level_func": "yellow",
130
- "level_lineno": "yellow",
131
- "level_message": "cyan",
132
- },
133
- }
134
- FORMAT = (
135
- FORMAT.replace("<white>", COLORS["white"])
136
- .replace("</white>", COLORS["reset"])
137
- .replace("<green>", COLORS["green"])
138
- .replace("</green>", COLORS["reset"])
139
- )
140
- LEVEL_FORMATS: dict[str, str] = {}
141
- for level, settings in LEVEL_SETTINGS.items():
142
- fmt = FORMAT
143
- for name, color in settings.items():
144
- fmt = fmt.replace(f"<{name}>", COLORS[color]).replace(f"</{name}>", COLORS["reset"])
145
- LEVEL_FORMATS[level] = fmt
146
-
147
- class TelegrinderLoggingFormatter(logging.Formatter):
148
- def format(self, record: logging.LogRecord) -> str:
149
- if not record.funcName or record.funcName == "<module>":
150
- record.funcName = "\b"
151
- frame = next(
152
- (
153
- frame
154
- for frame in inspect.stack()
155
- if frame.filename == record.pathname and frame.lineno == record.lineno
156
- ),
157
- None,
158
- )
159
- if frame:
160
- module = inspect.getmodule(frame.frame)
161
- record.module = module.__name__ if module else "<module>"
162
- return logging.Formatter(
163
- LEVEL_FORMATS.get(record.levelname),
164
- datefmt="%Y-%m-%d %H:%M:%S",
165
- style="{",
166
- ).format(record)
167
-
168
- class LogMessage:
169
- def __init__(self, fmt: typing.Any, args: typing.Any, kwargs: typing.Any) -> None:
170
- self.fmt = fmt
171
- self.args = args
172
- self.kwargs = kwargs
173
-
174
- def __str__(self) -> str:
175
- return self.fmt.format(*self.args, **self.kwargs)
176
-
177
- class TelegrinderLoggingStyleAdapter(logging.LoggerAdapter):
178
- def __init__(
179
- self,
180
- logger: LoggerModule,
181
- extra: dict[str, typing.Any] | None = None,
182
- ) -> None:
183
- super().__init__(logger, extra or {})
184
-
185
- def log(self, level: int, msg: object, *args: object, **kwargs: object) -> None:
186
- if self.isEnabledFor(level):
187
- kwargs.setdefault("stacklevel", 2)
188
- msg, args, kwargs = self.proc(msg, args, kwargs)
189
- self.logger._log(level, msg, args, **kwargs)
190
-
191
- def proc(
192
- self,
193
- msg: object,
194
- args: tuple[object, ...],
195
- kwargs: dict[str, object],
196
- ) -> tuple[LogMessage | object, tuple[object, ...], dict[str, object]]:
197
- log_kwargs = {
198
- key: kwargs[key] for key in inspect.getfullargspec(self.logger._log).args[1:] if key in kwargs
199
- }
200
-
201
- if isinstance(msg, str):
202
- msg = LogMessage(msg, args, kwargs)
203
- args = tuple()
204
- return msg, args, log_kwargs
205
-
206
- handler = logging.StreamHandler(sys.stderr)
207
- handler.setFormatter(TelegrinderLoggingFormatter())
208
- logger = logging.getLogger("telegrinder") # type: ignore
209
- logger.setLevel(logging_level) # type: ignore
210
- logger.addHandler(handler) # type: ignore
211
- logger = TelegrinderLoggingStyleAdapter(logger) # type: ignore
212
-
213
- if asyncio_module == "uvloop":
214
- import asyncio
215
-
216
- import uvloop # type: ignore
217
-
218
- asyncio.set_event_loop_policy(uvloop.EventLoopPolicy()) # type: ignore
219
-
220
-
221
- def _set_logger_level(level, /):
222
- level = level.upper()
223
- if logging_module == "logging":
224
- import logging
225
-
226
- logging.getLogger("telegrinder").setLevel(level)
227
- elif logging_module == "loguru":
228
- import loguru # type: ignore
229
-
230
- if handler_id in loguru.logger._core.handlers: # type: ignore
231
- loguru.logger._core.handlers[handler_id]._levelno = loguru.logger.level(level).no # type: ignore
232
-
233
-
234
- setattr(logger, "set_level", staticmethod(_set_logger_level)) # type: ignore
235
-
236
-
237
- __all__ = ("LoggerModule", "logger")
1
+ import os
2
+ import typing
3
+
4
+ from choicelib import choice_in_order
5
+
6
+ import telegrinder.msgspec_json as json
7
+
8
+
9
+ @typing.runtime_checkable
10
+ class LoggerModule(typing.Protocol):
11
+ def debug(self, __msg: object, *args: object, **kwargs: object) -> None: ...
12
+
13
+ def info(self, __msg: object, *args: object, **kwargs: object) -> None: ...
14
+
15
+ def warning(self, __msg: object, *args: object, **kwargs: object) -> None: ...
16
+
17
+ def error(self, __msg: object, *args: object, **kwargs: object) -> None: ...
18
+
19
+ def critical(self, __msg: object, *args: object, **kwargs: object) -> None: ...
20
+
21
+ def exception(self, __msg: object, *args: object, **kwargs: object) -> None: ...
22
+
23
+ if typing.TYPE_CHECKING:
24
+
25
+ def set_level(
26
+ self,
27
+ level: typing.Literal[
28
+ "DEBUG",
29
+ "INFO",
30
+ "WARNING",
31
+ "ERROR",
32
+ "CRITICAL",
33
+ "EXCEPTION",
34
+ ],
35
+ /,
36
+ ) -> None: ...
37
+
38
+
39
+ logger: LoggerModule
40
+ logging_level = os.getenv("LOGGER_LEVEL", default="DEBUG").upper()
41
+ logging_module = choice_in_order(["loguru"], default="logging", do_import=False)
42
+ asyncio_module = choice_in_order(["uvloop"], default="asyncio", do_import=False)
43
+
44
+ if logging_module == "loguru":
45
+ import os
46
+ import sys
47
+
48
+ from loguru import logger # type: ignore
49
+
50
+ os.environ.setdefault("LOGURU_AUTOINIT", "0")
51
+ log_format = (
52
+ "<level>{level: <8}</level> | "
53
+ "<lg>{time:YYYY-MM-DD HH:mm:ss}</lg> | "
54
+ "<le>{name}</le>:<le>{function}</le>:"
55
+ "<le>{line}</le> > <lw>{message}</lw>"
56
+ )
57
+ logger.remove() # type: ignore
58
+ handler_id = logger.add( # type: ignore
59
+ sink=sys.stderr,
60
+ format=log_format,
61
+ enqueue=True,
62
+ colorize=True,
63
+ level=logging_level,
64
+ )
65
+
66
+ elif logging_module == "logging":
67
+ """
68
+ This is a workaround for lazy formatting with {} in logging.
69
+ About:
70
+ https://docs.python.org/3/howto/logging-cookbook.html#use-of-alternative-formatting-styles
71
+ """
72
+
73
+ import inspect
74
+ import logging
75
+ import sys
76
+
77
+ import colorama
78
+
79
+ colorama.just_fix_windows_console() # init & fix console
80
+
81
+ FORMAT = (
82
+ "<white>{name: <4} |</white> <level>{levelname: <8}</level>"
83
+ " <white>|</white> <green>{asctime}</green> <white>|</white> <level_module>"
84
+ "{module}</level_module><white>:</white><level_func>"
85
+ "{funcName}</level_func><white>:</white><level_lineno>"
86
+ "{lineno}</level_lineno><white> > </white><level_message>"
87
+ "{message}</level_message>"
88
+ )
89
+ COLORS = {
90
+ "red": colorama.Fore.LIGHTRED_EX,
91
+ "green": colorama.Fore.LIGHTGREEN_EX,
92
+ "blue": colorama.Fore.LIGHTBLUE_EX,
93
+ "white": colorama.Fore.LIGHTWHITE_EX,
94
+ "yellow": colorama.Fore.LIGHTYELLOW_EX,
95
+ "magenta": colorama.Fore.LIGHTMAGENTA_EX,
96
+ "cyan": colorama.Fore.LIGHTCYAN_EX,
97
+ "reset": colorama.Style.RESET_ALL,
98
+ }
99
+ LEVEL_SETTINGS = {
100
+ "INFO": {
101
+ "level": "green",
102
+ "level_module": "blue",
103
+ "level_func": "cyan",
104
+ "level_lineno": "white",
105
+ "level_message": "green",
106
+ },
107
+ "DEBUG": {
108
+ "level": "blue",
109
+ "level_module": "yellow",
110
+ "level_func": "green",
111
+ "level_lineno": "cyan",
112
+ "level_message": "blue",
113
+ },
114
+ "WARNING": {
115
+ "level": "yellow",
116
+ "level_module": "red",
117
+ "level_func": "green",
118
+ "level_lineno": "red",
119
+ "level_message": "yellow",
120
+ },
121
+ "ERROR": {
122
+ "level": "red",
123
+ "level_module": "magenta",
124
+ "level_func": "yellow",
125
+ "level_lineno": "green",
126
+ "level_message": "red",
127
+ },
128
+ "CRITICAL": {
129
+ "level": "cyan",
130
+ "level_module": "yellow",
131
+ "level_func": "yellow",
132
+ "level_lineno": "yellow",
133
+ "level_message": "cyan",
134
+ },
135
+ }
136
+ FORMAT = (
137
+ FORMAT.replace("<white>", COLORS["white"])
138
+ .replace("</white>", COLORS["reset"])
139
+ .replace("<green>", COLORS["green"])
140
+ .replace("</green>", COLORS["reset"])
141
+ )
142
+ LEVEL_FORMATS: dict[str, str] = {}
143
+ for level, settings in LEVEL_SETTINGS.items():
144
+ fmt = FORMAT
145
+ for name, color in settings.items():
146
+ fmt = fmt.replace(f"<{name}>", COLORS[color]).replace(f"</{name}>", COLORS["reset"])
147
+ LEVEL_FORMATS[level] = fmt
148
+
149
+ class TelegrinderLoggingFormatter(logging.Formatter):
150
+ def format(self, record: logging.LogRecord) -> str:
151
+ if not record.funcName or record.funcName == "<module>":
152
+ record.funcName = "\b"
153
+ frame = next(
154
+ (
155
+ frame
156
+ for frame in inspect.stack()
157
+ if frame.filename == record.pathname and frame.lineno == record.lineno
158
+ ),
159
+ None,
160
+ )
161
+ if frame:
162
+ module = inspect.getmodule(frame.frame)
163
+ record.module = module.__name__ if module else "<module>"
164
+ return logging.Formatter(
165
+ LEVEL_FORMATS.get(record.levelname),
166
+ datefmt="%Y-%m-%d %H:%M:%S",
167
+ style="{",
168
+ ).format(record)
169
+
170
+ class LogMessage:
171
+ def __init__(self, fmt: typing.Any, args: typing.Any, kwargs: typing.Any) -> None:
172
+ self.fmt = fmt
173
+ self.args = args
174
+ self.kwargs = kwargs
175
+
176
+ def __str__(self) -> str:
177
+ return self.fmt.format(*self.args, **self.kwargs)
178
+
179
+ class TelegrinderLoggingStyleAdapter(logging.LoggerAdapter):
180
+ def __init__(
181
+ self,
182
+ logger: LoggerModule,
183
+ extra: dict[str, typing.Any] | None = None,
184
+ ) -> None:
185
+ super().__init__(logger, extra or {})
186
+
187
+ def log(self, level: int, msg: object, *args: object, **kwargs: object) -> None:
188
+ if self.isEnabledFor(level):
189
+ kwargs.setdefault("stacklevel", 2)
190
+ msg, args, kwargs = self.proc(msg, args, kwargs)
191
+ self.logger._log(level, msg, args, **kwargs)
192
+
193
+ def proc(
194
+ self,
195
+ msg: object,
196
+ args: tuple[object, ...],
197
+ kwargs: dict[str, object],
198
+ ) -> tuple[LogMessage | object, tuple[object, ...], dict[str, object]]:
199
+ log_kwargs = {
200
+ key: kwargs[key] for key in inspect.getfullargspec(self.logger._log).args[1:] if key in kwargs
201
+ }
202
+
203
+ if isinstance(msg, str):
204
+ msg = LogMessage(msg, args, kwargs)
205
+ args = tuple()
206
+ return msg, args, log_kwargs
207
+
208
+ handler = logging.StreamHandler(sys.stderr)
209
+ handler.setFormatter(TelegrinderLoggingFormatter())
210
+ logger = logging.getLogger("telegrinder") # type: ignore
211
+ logger.setLevel(logging_level) # type: ignore
212
+ logger.addHandler(handler) # type: ignore
213
+ logger = TelegrinderLoggingStyleAdapter(logger) # type: ignore
214
+
215
+ if asyncio_module == "uvloop":
216
+ import asyncio
217
+
218
+ import uvloop # type: ignore
219
+
220
+ asyncio.set_event_loop_policy(uvloop.EventLoopPolicy()) # type: ignore
221
+
222
+
223
+ def _set_logger_level(level, /):
224
+ level = level.upper()
225
+ if logging_module == "logging":
226
+ import logging
227
+
228
+ logging.getLogger("telegrinder").setLevel(level)
229
+ elif logging_module == "loguru":
230
+ import loguru # type: ignore
231
+
232
+ if handler_id in loguru.logger._core.handlers: # type: ignore
233
+ loguru.logger._core.handlers[handler_id]._levelno = loguru.logger.level(level).no # type: ignore
234
+
235
+
236
+ setattr(logger, "set_level", staticmethod(_set_logger_level)) # type: ignore
237
+
238
+
239
+ __all__ = ("LoggerModule", "json", "logger")
@@ -1,14 +1,14 @@
1
- import typing
2
-
3
- from telegrinder.msgspec_utils import decoder, encoder
4
-
5
-
6
- def loads(s: str | bytes) -> typing.Any:
7
- return decoder.decode(s)
8
-
9
-
10
- def dumps(o: typing.Any) -> str:
11
- return encoder.encode(o)
12
-
13
-
14
- __all__ = ("dumps", "loads")
1
+ import typing
2
+
3
+ from telegrinder.msgspec_utils import decoder, encoder
4
+
5
+
6
+ def loads(s: str | bytes) -> typing.Any:
7
+ return decoder.decode(s)
8
+
9
+
10
+ def dumps(o: typing.Any) -> str:
11
+ return encoder.encode(o)
12
+
13
+
14
+ __all__ = ("dumps", "loads")