telegrinder 0.1.dev170__tar.gz → 0.2.0__tar.gz

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-0.1.dev170 → telegrinder-0.2.0}/PKG-INFO +2 -2
  2. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/pyproject.toml +8 -8
  3. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/__init__.py +2 -2
  4. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/api/__init__.py +1 -2
  5. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/api/api.py +15 -6
  6. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/api/error.py +2 -1
  7. telegrinder-0.2.0/telegrinder/api/token.py +36 -0
  8. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/__init__.py +12 -6
  9. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/bot.py +18 -6
  10. telegrinder-0.2.0/telegrinder/bot/cute_types/__init__.py +17 -0
  11. telegrinder-0.2.0/telegrinder/bot/cute_types/base.py +235 -0
  12. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/cute_types/callback_query.py +10 -6
  13. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/cute_types/chat_join_request.py +4 -5
  14. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/cute_types/chat_member_updated.py +4 -6
  15. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/cute_types/inline_query.py +3 -4
  16. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/cute_types/message.py +32 -21
  17. telegrinder-0.2.0/telegrinder/bot/cute_types/update.py +74 -0
  18. telegrinder-0.2.0/telegrinder/bot/cute_types/utils.py +95 -0
  19. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/__init__.py +10 -11
  20. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/abc.py +8 -5
  21. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/context.py +17 -8
  22. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/dispatch.py +71 -48
  23. telegrinder-0.2.0/telegrinder/bot/dispatch/handler/__init__.py +5 -0
  24. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/handler/abc.py +4 -4
  25. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/handler/func.py +46 -22
  26. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/handler/message_reply.py +6 -7
  27. telegrinder-0.2.0/telegrinder/bot/dispatch/middleware/__init__.py +3 -0
  28. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/middleware/abc.py +2 -2
  29. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/process.py +38 -19
  30. telegrinder-0.2.0/telegrinder/bot/dispatch/return_manager/__init__.py +19 -0
  31. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/return_manager/abc.py +3 -3
  32. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/return_manager/callback_query.py +1 -2
  33. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/return_manager/inline_query.py +1 -2
  34. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/return_manager/message.py +1 -2
  35. telegrinder-0.2.0/telegrinder/bot/dispatch/view/__init__.py +22 -0
  36. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/view/abc.py +18 -16
  37. telegrinder-0.2.0/telegrinder/bot/dispatch/view/box.py +129 -0
  38. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/view/callback_query.py +1 -2
  39. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/view/chat_join_request.py +1 -2
  40. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/view/chat_member.py +16 -2
  41. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/view/inline_query.py +1 -2
  42. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/view/message.py +12 -5
  43. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/view/raw.py +9 -8
  44. telegrinder-0.2.0/telegrinder/bot/dispatch/waiter_machine/__init__.py +10 -0
  45. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/waiter_machine/machine.py +12 -8
  46. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/waiter_machine/middleware.py +1 -1
  47. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/dispatch/waiter_machine/short_state.py +4 -3
  48. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/polling/abc.py +1 -1
  49. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/polling/polling.py +6 -6
  50. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/__init__.py +20 -20
  51. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/abc.py +57 -43
  52. telegrinder-0.2.0/telegrinder/bot/rules/adapter/__init__.py +14 -0
  53. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/adapter/abc.py +6 -3
  54. telegrinder-0.2.0/telegrinder/bot/rules/adapter/errors.py +5 -0
  55. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/adapter/event.py +28 -13
  56. telegrinder-0.2.0/telegrinder/bot/rules/adapter/node.py +48 -0
  57. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/adapter/raw_update.py +13 -5
  58. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/callback_data.py +4 -4
  59. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/chat_join.py +4 -4
  60. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/command.py +5 -7
  61. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/func.py +2 -2
  62. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/fuzzy.py +1 -1
  63. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/inline.py +3 -3
  64. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/integer.py +1 -2
  65. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/markup.py +5 -3
  66. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/message_entities.py +2 -2
  67. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/node.py +2 -2
  68. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/regex.py +1 -1
  69. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/rule_enum.py +1 -1
  70. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/text.py +1 -2
  71. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/update.py +1 -2
  72. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/scenario/abc.py +2 -2
  73. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/scenario/checkbox.py +3 -4
  74. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/scenario/choice.py +1 -2
  75. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/model.py +89 -45
  76. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/modules.py +3 -3
  77. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/msgspec_utils.py +85 -57
  78. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/__init__.py +17 -10
  79. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/attachment.py +19 -16
  80. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/base.py +46 -22
  81. telegrinder-0.2.0/telegrinder/node/callback_query.py +14 -0
  82. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/command.py +6 -2
  83. telegrinder-0.2.0/telegrinder/node/composer.py +185 -0
  84. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/container.py +3 -3
  85. telegrinder-0.2.0/telegrinder/node/event.py +68 -0
  86. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/me.py +3 -0
  87. telegrinder-0.2.0/telegrinder/node/message.py +14 -0
  88. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/polymorphic.py +15 -10
  89. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/rule.py +20 -6
  90. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/scope.py +9 -1
  91. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/source.py +21 -11
  92. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/text.py +4 -4
  93. telegrinder-0.2.0/telegrinder/node/update.py +15 -0
  94. telegrinder-0.2.0/telegrinder/py.typed +0 -0
  95. telegrinder-0.2.0/telegrinder/rules.py +60 -0
  96. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/__init__.py +2 -2
  97. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/buttons.py +5 -10
  98. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/error_handler/abc.py +2 -2
  99. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/error_handler/error.py +2 -0
  100. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/error_handler/error_handler.py +6 -6
  101. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/formatting/spec_html_formats.py +10 -10
  102. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/global_context/__init__.py +2 -2
  103. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/global_context/global_context.py +3 -3
  104. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/global_context/telegrinder_ctx.py +4 -4
  105. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/keyboard.py +3 -3
  106. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/loop_wrapper/loop_wrapper.py +47 -13
  107. telegrinder-0.2.0/telegrinder/tools/magic.py +166 -0
  108. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/types/__init__.py +1 -0
  109. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/types/enums.py +2 -0
  110. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/types/methods.py +91 -15
  111. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/types/objects.py +49 -24
  112. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/verification_utils.py +1 -3
  113. telegrinder-0.1.dev170/telegrinder/api/abc.py +0 -73
  114. telegrinder-0.1.dev170/telegrinder/bot/cute_types/__init__.py +0 -17
  115. telegrinder-0.1.dev170/telegrinder/bot/cute_types/base.py +0 -133
  116. telegrinder-0.1.dev170/telegrinder/bot/cute_types/update.py +0 -27
  117. telegrinder-0.1.dev170/telegrinder/bot/cute_types/utils.py +0 -558
  118. telegrinder-0.1.dev170/telegrinder/bot/dispatch/handler/__init__.py +0 -5
  119. telegrinder-0.1.dev170/telegrinder/bot/dispatch/middleware/__init__.py +0 -3
  120. telegrinder-0.1.dev170/telegrinder/bot/dispatch/return_manager/__init__.py +0 -19
  121. telegrinder-0.1.dev170/telegrinder/bot/dispatch/view/__init__.py +0 -22
  122. telegrinder-0.1.dev170/telegrinder/bot/dispatch/view/box.py +0 -118
  123. telegrinder-0.1.dev170/telegrinder/bot/dispatch/waiter_machine/__init__.py +0 -10
  124. telegrinder-0.1.dev170/telegrinder/bot/rules/adapter/__init__.py +0 -14
  125. telegrinder-0.1.dev170/telegrinder/bot/rules/adapter/errors.py +0 -4
  126. telegrinder-0.1.dev170/telegrinder/bot/rules/adapter/node.py +0 -42
  127. telegrinder-0.1.dev170/telegrinder/node/callback_query.py +0 -18
  128. telegrinder-0.1.dev170/telegrinder/node/composer.py +0 -160
  129. telegrinder-0.1.dev170/telegrinder/node/message.py +0 -18
  130. telegrinder-0.1.dev170/telegrinder/node/update.py +0 -12
  131. telegrinder-0.1.dev170/telegrinder/rules.py +0 -1
  132. telegrinder-0.1.dev170/telegrinder/tools/magic.py +0 -88
  133. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/LICENSE +0 -0
  134. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/readme.md +0 -0
  135. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/api/response.py +0 -0
  136. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/polling/__init__.py +0 -0
  137. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/enum_text.py +0 -0
  138. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/is_from.py +0 -0
  139. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/mention.py +0 -0
  140. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/message.py +0 -0
  141. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/rules/start.py +0 -0
  142. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/bot/scenario/__init__.py +0 -0
  143. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/client/__init__.py +0 -0
  144. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/client/abc.py +0 -0
  145. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/client/aiohttp.py +0 -0
  146. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/msgspec_json.py +0 -0
  147. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/tools/__init__.py +0 -0
  148. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/node/tools/generator.py +0 -0
  149. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/error_handler/__init__.py +0 -0
  150. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/formatting/__init__.py +0 -0
  151. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/formatting/html.py +0 -0
  152. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/formatting/links.py +0 -0
  153. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/global_context/abc.py +0 -0
  154. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/i18n/__init__.py +0 -0
  155. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/i18n/base.py +0 -0
  156. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/i18n/middleware/__init__.py +0 -0
  157. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/i18n/middleware/base.py +0 -0
  158. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/i18n/simple.py +0 -0
  159. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/kb_set/__init__.py +0 -0
  160. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/kb_set/base.py +0 -0
  161. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/kb_set/yaml.py +0 -0
  162. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/limited_dict.py +0 -0
  163. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/loop_wrapper/__init__.py +0 -0
  164. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/loop_wrapper/abc.py +0 -0
  165. {telegrinder-0.1.dev170 → telegrinder-0.2.0}/telegrinder/tools/parse_mode.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: telegrinder
3
- Version: 0.1.dev170
3
+ Version: 0.2.0
4
4
  Summary: Modern visionary telegram bot framework.
5
5
  Home-page: https://github.com/timoniq/telegrinder
6
6
  License: MIT
@@ -25,7 +25,7 @@ Requires-Dist: certifi (>=2024.2.2,<2025.0.0)
25
25
  Requires-Dist: choicelib (>=0.1.5,<0.2.0)
26
26
  Requires-Dist: colorama (>=0.4.0,<0.5.0)
27
27
  Requires-Dist: envparse (>=0.2.0,<0.3.0)
28
- Requires-Dist: fntypes (>=0.1.2.post2,<0.2.0)
28
+ Requires-Dist: fntypes (>=0.1.3,<0.2.0)
29
29
  Requires-Dist: msgspec (>=0.18.6,<0.19.0)
30
30
  Requires-Dist: requests (>=2.28.1,<3.0.0)
31
31
  Requires-Dist: typing-extensions (>=4.10.0,<5.0.0)
@@ -1,6 +1,6 @@
1
1
  [tool.poetry]
2
2
  name = "telegrinder"
3
- version = "0.1.dev170"
3
+ version = "0.2.0"
4
4
  description = "Modern visionary telegram bot framework."
5
5
  authors = ["timoniq <tesseradecades@mail.ru>"]
6
6
  maintainers = ["luwqz1 <howluwqz1@gmail.com>"]
@@ -29,6 +29,7 @@ classifiers = [
29
29
  "Intended Audience :: Developers",
30
30
  "License :: OSI Approved :: MIT License",
31
31
  "Programming Language :: Python :: 3.11",
32
+ "Programming Language :: Python :: 3.12",
32
33
  "Topic :: Software Development :: Libraries :: Python Modules",
33
34
  "Topic :: Software Development :: Quality Assurance",
34
35
  "Typing :: Typed",
@@ -49,10 +50,11 @@ certifi = "^2024.2.2"
49
50
  msgspec = "^0.18.6"
50
51
  requests = "^2.28.1"
51
52
  typing-extensions = "^4.10.0"
52
- fntypes = "^0.1.2.post2"
53
+ fntypes = "^0.1.3"
53
54
 
54
55
  [tool.poetry.group.dev.dependencies]
55
- ruff = "^0.5.0"
56
+ pre-commit = "^3.8.0"
57
+ ruff = ">=0.5,<0.7"
56
58
  basedpyright = "^1.12.1"
57
59
  sort-all = "1.2.0"
58
60
  pytest = "^8.0.0"
@@ -85,11 +87,9 @@ docstring-quotes = "double"
85
87
 
86
88
  [tool.pyright]
87
89
  exclude = [
88
- "**/__pycache__",
89
- "examples/blueprint_bot",
90
- "examples/webhook_bot",
91
- "docs",
92
- "local",
90
+ "**/__pycache__",
91
+ "docs",
92
+ "local",
93
93
  ]
94
94
  typeCheckingMode = "basic"
95
95
  pythonPlatform = "All"
@@ -34,7 +34,7 @@ bot.run_forever()
34
34
 
35
35
  import typing
36
36
 
37
- from .api import ABCAPI, API, APIError, APIResponse, Token
37
+ from .api import API, APIError, APIResponse, Token
38
38
  from .bot import (
39
39
  ABCDispatch,
40
40
  ABCHandler,
@@ -122,7 +122,7 @@ Bot: typing.TypeAlias = Telegrinder
122
122
 
123
123
 
124
124
  __all__ = (
125
- "ABCAPI",
125
+ "API",
126
126
  "ABCClient",
127
127
  "ABCDispatch",
128
128
  "ABCErrorHandler",
@@ -1,10 +1,9 @@
1
- from .abc import ABCAPI, Token
2
1
  from .api import API
3
2
  from .error import APIError, InvalidTokenError
4
3
  from .response import APIResponse
4
+ from .token import Token
5
5
 
6
6
  __all__ = (
7
- "ABCAPI",
8
7
  "API",
9
8
  "APIError",
10
9
  "APIResponse",
@@ -1,32 +1,34 @@
1
1
  import typing
2
+ from functools import cached_property
2
3
 
3
4
  import msgspec
4
5
  from fntypes.result import Error, Ok, Result
5
6
 
7
+ from telegrinder.api.error import APIError
6
8
  from telegrinder.api.response import APIResponse
9
+ from telegrinder.api.token import Token
7
10
  from telegrinder.client import ABCClient, AiohttpClient
8
11
  from telegrinder.model import DataConverter, decoder
9
12
  from telegrinder.types.methods import APIMethods
10
13
 
11
- from .abc import ABCAPI, APIError, Token
12
-
13
14
 
14
15
  def compose_data(
15
16
  client: ABCClient,
16
17
  data: dict[str, typing.Any],
17
18
  files: dict[str, tuple[str, bytes]],
18
19
  ) -> typing.Any:
19
- converter = DataConverter(files=files.copy())
20
+ converter = DataConverter(_files=files.copy())
20
21
  return client.get_form(
21
22
  data={k: converter(v) for k, v in data.items()},
22
23
  files=converter.files,
23
24
  )
24
25
 
25
26
 
26
- class API(ABCAPI, APIMethods):
27
- """Bot API with available API methods."""
27
+ class API(APIMethods):
28
+ """Bot API with available API methods and http client."""
28
29
 
29
30
  API_URL = "https://api.telegram.org/"
31
+ API_FILE_URL = "https://api.telegram.org/file/"
30
32
 
31
33
  def __init__(self, token: Token, *, http: ABCClient | None = None) -> None:
32
34
  self.token = token
@@ -40,7 +42,7 @@ class API(ABCAPI, APIMethods):
40
42
  self.http,
41
43
  )
42
44
 
43
- @property
45
+ @cached_property
44
46
  def id(self) -> int:
45
47
  return self.token.bot_id
46
48
 
@@ -48,6 +50,13 @@ class API(ABCAPI, APIMethods):
48
50
  def request_url(self) -> str:
49
51
  return self.API_URL + f"bot{self.token}/"
50
52
 
53
+ @property
54
+ def request_file_url(self) -> str:
55
+ return self.API_FILE_URL + f"bot{self.token}/"
56
+
57
+ async def download_file(self, file_path: str) -> bytes:
58
+ return await self.http.request_content(f"{self.request_file_url}/{file_path}")
59
+
51
60
  async def request(
52
61
  self,
53
62
  method: str,
@@ -9,7 +9,8 @@ class APIError(BaseException):
9
9
  return f"<APIError: {self.__str__()}>"
10
10
 
11
11
 
12
- class InvalidTokenError(BaseException): ...
12
+ class InvalidTokenError(BaseException):
13
+ pass
13
14
 
14
15
 
15
16
  __all__ = ("APIError", "InvalidTokenError")
@@ -0,0 +1,36 @@
1
+ import pathlib
2
+ import typing
3
+ from functools import cached_property
4
+
5
+ from envparse import env
6
+
7
+ from .error import InvalidTokenError
8
+
9
+
10
+ class Token(str):
11
+ def __new__(cls, token: str) -> typing.Self:
12
+ if token.count(":") != 1 or not token.split(":")[0].isdigit():
13
+ raise InvalidTokenError("Invalid token, it should look like this '123:ABC'.")
14
+ return super().__new__(cls, token)
15
+
16
+ def __repr__(self) -> str:
17
+ return f"<Token: {self.bot_id}:{''.join(self.split(':')[-1])[:6]}...>"
18
+
19
+ @classmethod
20
+ def from_env(
21
+ cls,
22
+ var_name: str = "BOT_TOKEN",
23
+ *,
24
+ is_read: bool = False,
25
+ path_to_envfile: str | pathlib.Path | None = None,
26
+ ) -> typing.Self:
27
+ if not is_read:
28
+ env.read_envfile(path_to_envfile)
29
+ return cls(env.str(var_name))
30
+
31
+ @cached_property
32
+ def bot_id(self) -> int:
33
+ return int(self.split(":")[0])
34
+
35
+
36
+ __all__ = ("Token",)
@@ -1,5 +1,5 @@
1
- from .bot import Telegrinder
2
- from .cute_types import (
1
+ from telegrinder.bot.bot import Telegrinder
2
+ from telegrinder.bot.cute_types import (
3
3
  BaseCute,
4
4
  CallbackQueryCute,
5
5
  ChatJoinRequestCute,
@@ -8,7 +8,7 @@ from .cute_types import (
8
8
  MessageCute,
9
9
  UpdateCute,
10
10
  )
11
- from .dispatch import (
11
+ from telegrinder.bot.dispatch import (
12
12
  ABCDispatch,
13
13
  ABCHandler,
14
14
  ABCMiddleware,
@@ -37,9 +37,15 @@ from .dispatch import (
37
37
  clear_wm_storage_worker,
38
38
  register_manager,
39
39
  )
40
- from .polling import ABCPolling, Polling
41
- from .rules import ABCRule, CallbackQueryRule, ChatJoinRequestRule, InlineQueryRule, MessageRule
42
- from .scenario import ABCScenario, Checkbox, Choice
40
+ from telegrinder.bot.polling import ABCPolling, Polling
41
+ from telegrinder.bot.rules import (
42
+ ABCRule,
43
+ CallbackQueryRule,
44
+ ChatJoinRequestRule,
45
+ InlineQueryRule,
46
+ MessageRule,
47
+ )
48
+ from telegrinder.bot.scenario import ABCScenario, Checkbox, Choice
43
49
 
44
50
  __all__ = (
45
51
  "ABCDispatch",
@@ -1,10 +1,13 @@
1
1
  import typing_extensions as typing
2
2
 
3
- from telegrinder.api import API
4
- from telegrinder.bot.dispatch import ABCDispatch, Dispatch
5
- from telegrinder.bot.polling import ABCPolling, Polling
3
+ from telegrinder.api.api import API
4
+ from telegrinder.bot.dispatch.abc import ABCDispatch
5
+ from telegrinder.bot.dispatch.dispatch import Dispatch
6
+ from telegrinder.bot.polling.abc import ABCPolling
7
+ from telegrinder.bot.polling.polling import Polling
6
8
  from telegrinder.modules import logger
7
- from telegrinder.tools.loop_wrapper import ABCLoopWrapper, LoopWrapper
9
+ from telegrinder.tools.loop_wrapper import ABCLoopWrapper
10
+ from telegrinder.tools.loop_wrapper.loop_wrapper import LoopWrapper
8
11
 
9
12
  DispatchT = typing.TypeVar("DispatchT", bound=ABCDispatch, default=Dispatch)
10
13
  PollingT = typing.TypeVar("PollingT", bound=ABCPolling, default=Polling)
@@ -43,7 +46,12 @@ class Telegrinder(typing.Generic[DispatchT, PollingT, LoopWrapperT]):
43
46
  return
44
47
  await self.api.delete_webhook()
45
48
 
46
- async def run_polling(self, *, offset: int = 0, skip_updates: bool = False) -> None:
49
+ async def run_polling(
50
+ self,
51
+ *,
52
+ offset: int = 0,
53
+ skip_updates: bool = False,
54
+ ) -> None:
47
55
  if skip_updates:
48
56
  logger.debug("Dropping pending updates")
49
57
  await self.reset_webhook()
@@ -52,7 +60,11 @@ class Telegrinder(typing.Generic[DispatchT, PollingT, LoopWrapperT]):
52
60
 
53
61
  async for updates in self.polling.listen():
54
62
  for update in updates:
55
- logger.debug("Received update (update_id={})", update.update_id)
63
+ logger.debug(
64
+ "Received update (update_id={}, update_type={!r})",
65
+ update.update_id,
66
+ update.update_type.name,
67
+ )
56
68
  self.loop_wrapper.add_task(self.dispatch.feed(update, self.api))
57
69
 
58
70
  def run_forever(self, *, offset: int = 0, skip_updates: bool = False) -> None:
@@ -0,0 +1,17 @@
1
+ from telegrinder.bot.cute_types.base import BaseCute
2
+ from telegrinder.bot.cute_types.callback_query import CallbackQueryCute
3
+ from telegrinder.bot.cute_types.chat_join_request import ChatJoinRequestCute
4
+ from telegrinder.bot.cute_types.chat_member_updated import ChatMemberUpdatedCute
5
+ from telegrinder.bot.cute_types.inline_query import InlineQueryCute
6
+ from telegrinder.bot.cute_types.message import MessageCute
7
+ from telegrinder.bot.cute_types.update import UpdateCute
8
+
9
+ __all__ = (
10
+ "BaseCute",
11
+ "CallbackQueryCute",
12
+ "ChatJoinRequestCute",
13
+ "ChatMemberUpdatedCute",
14
+ "InlineQueryCute",
15
+ "MessageCute",
16
+ "UpdateCute",
17
+ )
@@ -0,0 +1,235 @@
1
+ import dataclasses
2
+ import inspect
3
+ import typing
4
+ from functools import wraps
5
+
6
+ import typing_extensions
7
+ from fntypes.result import Result
8
+
9
+ from telegrinder.api.api import API
10
+ from telegrinder.model import Model, get_params
11
+
12
+ F = typing.TypeVar("F", bound=typing.Callable[..., typing.Any])
13
+ Cute = typing.TypeVar("Cute", bound="BaseCute")
14
+ Update = typing_extensions.TypeVar("Update", bound=Model)
15
+ CtxAPI = typing_extensions.TypeVar("CtxAPI", bound=API, default=API)
16
+
17
+ Executor: typing.TypeAlias = typing.Callable[
18
+ [Cute, str, dict[str, typing.Any]],
19
+ typing.Awaitable[Result[typing.Any, typing.Any]],
20
+ ]
21
+
22
+ if typing.TYPE_CHECKING:
23
+
24
+ class BaseCute(Model, typing.Generic[Update, CtxAPI]):
25
+ api: API
26
+
27
+ @classmethod
28
+ def from_update(cls, update: Update, bound_api: API) -> typing.Self: ...
29
+
30
+ @property
31
+ def ctx_api(self) -> CtxAPI: ...
32
+
33
+ def to_dict(
34
+ self,
35
+ *,
36
+ exclude_fields: set[str] | None = None,
37
+ ) -> dict[str, typing.Any]:
38
+ """
39
+ :param exclude_fields: Cute model field names to exclude from the dictionary representation of this cute model.
40
+ :return: A dictionary representation of this cute model.
41
+ """
42
+
43
+ ...
44
+
45
+ def to_full_dict(
46
+ self,
47
+ *,
48
+ exclude_fields: set[str] | None = None,
49
+ ) -> dict[str, typing.Any]:
50
+ """
51
+ :param exclude_fields: Cute model field names to exclude from the dictionary representation of this cute model.
52
+ :return: A dictionary representation of this model including all models, structs, custom types.
53
+ """
54
+
55
+ ...
56
+
57
+ else:
58
+ from fntypes.co import Some, Variative
59
+
60
+ from telegrinder.msgspec_utils import Option, decoder
61
+ from telegrinder.msgspec_utils import get_class_annotations as _get_class_annotations
62
+
63
+ def _get_cute_from_args(args):
64
+ for hint in args:
65
+ if not isinstance(hint, type):
66
+ hint = typing.get_origin(hint) or hint
67
+ if not isinstance(hint, type):
68
+ continue
69
+
70
+ if hint in (Variative, Some, Option):
71
+ return _get_cute_from_args(typing.get_args(hint))
72
+ if issubclass(hint, BaseCute):
73
+ return hint
74
+
75
+ return None
76
+
77
+ def _get_cute_annotations(annotations):
78
+ cute_annotations = {}
79
+
80
+ for key, hint in annotations.items():
81
+ if not isinstance(hint, type):
82
+ if (val := _get_cute_from_args(typing.get_args(hint))) is not None:
83
+ cute_annotations[key] = val
84
+
85
+ elif issubclass(hint, BaseCute):
86
+ cute_annotations[key] = hint
87
+
88
+ return cute_annotations
89
+
90
+ def _get_value(value):
91
+ while isinstance(value, Variative | Some):
92
+ if isinstance(value, Variative):
93
+ value = value.v
94
+ if isinstance(value, Some):
95
+ value = value.value
96
+ return value
97
+
98
+ class BaseCute(typing.Generic[Update, CtxAPI]):
99
+ def __init_subclass__(cls, *args, **kwargs):
100
+ super().__init_subclass__(*args, **kwargs)
101
+
102
+ if not cls.__bases__ or not issubclass(cls.__bases__[0], BaseCute):
103
+ return
104
+
105
+ cls.__is_solved_annotations__ = False
106
+ cls.__cute_annotations__ = None
107
+
108
+ @classmethod
109
+ def from_update(cls, update, bound_api):
110
+ if not cls.__is_solved_annotations__:
111
+ cls.__is_solved_annotations__ = True
112
+ cls.__annotations__ = _get_class_annotations(cls)
113
+
114
+ if cls.__cute_annotations__ is None:
115
+ cls.__cute_annotations__ = _get_cute_annotations(cls.__annotations__)
116
+
117
+ return cls(
118
+ **{
119
+ field: decoder.convert(
120
+ cls.__cute_annotations__[field].from_update(_get_value(value), bound_api=bound_api),
121
+ type=cls.__annotations__[field],
122
+ )
123
+ if field in cls.__cute_annotations__ and value
124
+ else value
125
+ for field, value in update.to_dict().items()
126
+ },
127
+ api=bound_api,
128
+ )
129
+
130
+ @property
131
+ def ctx_api(self):
132
+ return self.api
133
+
134
+ def to_dict(self, *, exclude_fields=None):
135
+ exclude_fields = exclude_fields or set()
136
+ return super().to_dict(exclude_fields={"api"} | exclude_fields)
137
+
138
+ def to_full_dict(self, *, exclude_fields=None):
139
+ exclude_fields = exclude_fields or set()
140
+ return super().to_full_dict(exclude_fields={"api"} | exclude_fields)
141
+
142
+
143
+ def compose_method_params(
144
+ params: dict[str, typing.Any],
145
+ update: Cute,
146
+ *,
147
+ default_params: set[str | tuple[str, str]] | None = None,
148
+ validators: dict[str, typing.Callable[[Cute], bool]] | None = None,
149
+ ) -> dict[str, typing.Any]:
150
+ """Compose method `params` from `update` by `default_params` and `validators`.
151
+
152
+ :param params: Method params.
153
+ :param update: Update object.
154
+ :param default_params: Default params. \
155
+ (`str`) - Attribute name to be get from `update` if param is undefined. \
156
+ (`tuple[str, str]`): tuple[0] - Parameter name to be set in `params`, \
157
+ tuple[1] - attribute name to be get from `update`.
158
+ :param validators: Validators mapping (`str, Callable`), key - `Parameter name` \
159
+ for which the validator will be applied, value - `Validator`, if returned `True` \
160
+ parameter will be set, otherwise will not.
161
+ :return: Composed params.
162
+ """
163
+
164
+ default_params = default_params or set()
165
+ validators = validators or {}
166
+
167
+ for param in default_params:
168
+ param_name = param if isinstance(param, str) else param[0]
169
+ if param_name not in params:
170
+ if param_name in validators and not validators[param_name](update):
171
+ continue
172
+ params[param_name] = getattr(update, param if isinstance(param, str) else param[1])
173
+
174
+ return params
175
+
176
+
177
+ def shortcut(
178
+ method_name: str,
179
+ *,
180
+ executor: Executor[Cute] | None = None,
181
+ custom_params: set[str] | None = None,
182
+ ):
183
+ def wrapper(func: F) -> F:
184
+ @wraps(func)
185
+ async def inner(
186
+ self: Cute,
187
+ *args: typing.Any,
188
+ **kwargs: typing.Any,
189
+ ) -> typing.Any:
190
+ if executor is None:
191
+ return await func(self, *args, **kwargs)
192
+
193
+ if not hasattr(func, "_signature_params"):
194
+ setattr(
195
+ func,
196
+ "_signature_params",
197
+ {k: p for k, p in inspect.signature(func).parameters.items() if k != "self"},
198
+ )
199
+
200
+ signature_params: dict[str, inspect.Parameter] = getattr(func, "_signature_params")
201
+ params: dict[str, typing.Any] = {}
202
+ index = 0
203
+
204
+ for k, p in signature_params.items():
205
+ if p.kind in (p.POSITIONAL_OR_KEYWORD, p.POSITIONAL_ONLY) and len(args) > index:
206
+ params[k] = args[index]
207
+ index += 1
208
+ continue
209
+
210
+ if p.kind in (p.VAR_KEYWORD, p.VAR_POSITIONAL):
211
+ params[k] = kwargs.copy() if p.kind is p.VAR_KEYWORD else args[index:]
212
+ continue
213
+
214
+ params[k] = kwargs.pop(k, p.default) if p.default is not p.empty else kwargs.pop(k)
215
+
216
+ return await executor(self, method_name, get_params(params))
217
+
218
+ inner.__shortcut__ = Shortcut( # type: ignore
219
+ method_name=method_name,
220
+ executor=executor,
221
+ custom_params=custom_params or set(),
222
+ )
223
+ return inner # type: ignore
224
+
225
+ return wrapper
226
+
227
+
228
+ @dataclasses.dataclass(slots=True, frozen=True)
229
+ class Shortcut:
230
+ method_name: str
231
+ executor: Executor | None = dataclasses.field(default=None, kw_only=True)
232
+ custom_params: set[str] = dataclasses.field(default_factory=lambda: set(), kw_only=True)
233
+
234
+
235
+ __all__ = ("BaseCute", "Shortcut", "compose_method_params", "shortcut")
@@ -4,12 +4,15 @@ from contextlib import suppress
4
4
  import msgspec
5
5
  from fntypes.co import Nothing, Result, Some, Variative, unwrapping
6
6
 
7
- from telegrinder.api import ABCAPI, APIError
7
+ from telegrinder.api import API, APIError
8
+ from telegrinder.bot.cute_types.base import BaseCute, compose_method_params, shortcut
9
+ from telegrinder.bot.cute_types.message import MediaType, MessageCute, ReplyMarkup, execute_method_edit
8
10
  from telegrinder.model import get_params
9
11
  from telegrinder.msgspec_utils import Option, decoder
10
- from telegrinder.types import (
12
+ from telegrinder.types.objects import (
11
13
  CallbackQuery,
12
14
  Chat,
15
+ InaccessibleMessage,
13
16
  InlineKeyboardMarkup,
14
17
  InputFile,
15
18
  InputMedia,
@@ -20,12 +23,13 @@ from telegrinder.types import (
20
23
  User,
21
24
  )
22
25
 
23
- from .base import BaseCute, compose_method_params, shortcut
24
- from .message import MediaType, MessageCute, ReplyMarkup, execute_method_edit
25
26
 
27
+ class CallbackQueryCute(BaseCute[CallbackQuery], CallbackQuery, kw_only=True):
28
+ api: API
26
29
 
27
- class CallbackQueryCute(BaseCute[CallbackQuery], CallbackQuery, kw_only=True, dict=True):
28
- api: ABCAPI
30
+ message: Option[Variative[MessageCute, InaccessibleMessage]] = Nothing()
31
+ """Optional. Message sent by the bot with the callback button that originated
32
+ the query."""
29
33
 
30
34
  @property
31
35
  def from_user(self) -> User:
@@ -2,15 +2,14 @@ import typing
2
2
 
3
3
  from fntypes.result import Result
4
4
 
5
- from telegrinder.api.abc import ABCAPI, APIError
5
+ from telegrinder.api import API, APIError
6
+ from telegrinder.bot.cute_types.base import BaseCute, shortcut
7
+ from telegrinder.bot.cute_types.chat_member_updated import ChatMemberShortcuts, chat_member_interaction
6
8
  from telegrinder.types.objects import ChatJoinRequest, User
7
9
 
8
- from .base import BaseCute, shortcut
9
- from .chat_member_updated import ChatMemberShortcuts, chat_member_interaction
10
-
11
10
 
12
11
  class ChatJoinRequestCute(BaseCute[ChatJoinRequest], ChatJoinRequest, ChatMemberShortcuts, kw_only=True):
13
- api: ABCAPI
12
+ api: API
14
13
 
15
14
  @property
16
15
  def from_user(self) -> User:
@@ -3,13 +3,11 @@ from datetime import datetime
3
3
 
4
4
  from fntypes.result import Result
5
5
 
6
- from telegrinder.api import ABCAPI, APIError
6
+ from telegrinder.api import API, APIError
7
+ from telegrinder.bot.cute_types.base import BaseCute, compose_method_params, shortcut
7
8
  from telegrinder.model import get_params
8
9
  from telegrinder.types.objects import ChatMemberUpdated, ChatPermissions, User
9
10
 
10
- from .base import BaseCute, compose_method_params, shortcut
11
- from .utils import compose_chat_permissions
12
-
13
11
 
14
12
  async def chat_member_interaction(
15
13
  update: BaseCute[typing.Any],
@@ -17,7 +15,7 @@ async def chat_member_interaction(
17
15
  params: dict[str, typing.Any],
18
16
  ) -> Result[typing.Any, APIError]:
19
17
  if isinstance(params.get("permissions"), dict):
20
- params["permissions"] = compose_chat_permissions(**params["permissions"])
18
+ params["permissions"] = ChatPermissions(**params["permissions"])
21
19
  params = compose_method_params(
22
20
  get_params(locals()),
23
21
  update,
@@ -235,7 +233,7 @@ class ChatMemberShortcuts:
235
233
  class ChatMemberUpdatedCute(
236
234
  BaseCute[ChatMemberUpdated], ChatMemberUpdated, ChatMemberShortcuts, kw_only=True
237
235
  ):
238
- api: ABCAPI
236
+ api: API
239
237
 
240
238
  @property
241
239
  def from_user(self) -> User:
@@ -2,7 +2,8 @@ import typing
2
2
 
3
3
  from fntypes.result import Result
4
4
 
5
- from telegrinder.api import ABCAPI, APIError
5
+ from telegrinder.api import API, APIError
6
+ from telegrinder.bot.cute_types.base import BaseCute, compose_method_params, shortcut
6
7
  from telegrinder.model import get_params
7
8
  from telegrinder.types import (
8
9
  InlineQuery,
@@ -11,11 +12,9 @@ from telegrinder.types import (
11
12
  User,
12
13
  )
13
14
 
14
- from .base import BaseCute, compose_method_params, shortcut
15
-
16
15
 
17
16
  class InlineQueryCute(BaseCute[InlineQuery], InlineQuery, kw_only=True):
18
- api: ABCAPI
17
+ api: API
19
18
 
20
19
  @property
21
20
  def from_user(self) -> User: