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
@@ -1,77 +1,76 @@
1
- import importlib.util as importlib_util
2
- import os
3
- import pathlib
4
- import sys
5
- import typing
6
- from abc import ABC, abstractmethod
7
-
8
- from fntypes.option import Option
9
-
10
- from telegrinder.api.api import API
11
- from telegrinder.tools.global_context.abc import ABCGlobalContext
12
- from telegrinder.types.objects import Update
13
-
14
- T = typing.TypeVar("T")
15
-
16
-
17
- class PathExistsError(BaseException):
18
- pass
19
-
20
-
21
- class ABCDispatch(ABC):
22
- @property
23
- @abstractmethod
24
- def global_context(self) -> ABCGlobalContext:
25
- pass
26
-
27
- @abstractmethod
28
- async def feed(self, event: Update, api: API) -> bool:
29
- pass
30
-
31
- @abstractmethod
32
- def load(self, external: typing.Self) -> None:
33
- pass
34
-
35
- @abstractmethod
36
- def get_view(self, of_type: type[T]) -> Option[T]:
37
- pass
38
-
39
- def load_many(self, *externals: typing.Self) -> None:
40
- for external in externals:
41
- self.load(external)
42
-
43
- def load_from_dir(self, directory: str | pathlib.Path) -> bool:
44
- """Loads dispatchers from a directory containing Python modules where global variables
45
- are declared with instances of dispatch.
46
- Returns True if dispatchers were found, otherwise False."""
47
-
48
- directory = pathlib.Path(directory)
49
-
50
- if not directory.exists():
51
- raise PathExistsError(f"Path {str(directory)!r} does not exists.")
52
-
53
- dps: list[typing.Self] = []
54
- for root, _, files in os.walk(directory):
55
- for f in files:
56
- if f.endswith(".py") and f != "__init__.py":
57
- module_path = os.path.join(root, f)
58
- module_name = os.path.splitext(os.path.relpath(module_path, directory))[0]
59
- module_name = module_name.replace(os.sep, ".")
60
-
61
- spec = importlib_util.spec_from_file_location(module_name, module_path)
62
- if spec is None or spec.loader is None:
63
- continue
64
-
65
- module = importlib_util.module_from_spec(spec)
66
- sys.modules[module_name] = module
67
- spec.loader.exec_module(module)
68
-
69
- for obj in module.__dict__.values():
70
- if isinstance(obj, self.__class__):
71
- dps.append(obj)
72
-
73
- self.load_many(*dps)
74
- return bool(dps)
75
-
76
-
77
- __all__ = ("ABCDispatch",)
1
+ import importlib.util as importlib_util
2
+ import os
3
+ import pathlib
4
+ import sys
5
+ import typing
6
+ from abc import ABC, abstractmethod
7
+
8
+ from fntypes.option import Option
9
+
10
+ from telegrinder.api.api import API
11
+ from telegrinder.types.objects import Update
12
+
13
+ if typing.TYPE_CHECKING:
14
+ from telegrinder.bot.dispatch.view.abc import ABCView
15
+
16
+
17
+ class PathExistsError(BaseException):
18
+ pass
19
+
20
+
21
+ class ABCDispatch(ABC):
22
+ @abstractmethod
23
+ async def feed(self, event: Update, api: API[typing.Any]) -> bool:
24
+ pass
25
+
26
+ @abstractmethod
27
+ def load(self, external: typing.Self) -> None:
28
+ pass
29
+
30
+ @abstractmethod
31
+ def get_view[T](self, of_type: type[T]) -> Option[T]:
32
+ pass
33
+
34
+ @abstractmethod
35
+ def get_views(self) -> dict[str, "ABCView"]:
36
+ pass
37
+
38
+ def load_many(self, *externals: typing.Self) -> None:
39
+ for external in externals:
40
+ self.load(external)
41
+
42
+ def load_from_dir(self, directory: str | pathlib.Path) -> bool:
43
+ """Loads dispatchers from a directory containing Python modules where global variables
44
+ are declared with instances of dispatch.
45
+ Returns True if dispatchers were found, otherwise False.
46
+ """
47
+ directory = pathlib.Path(directory)
48
+
49
+ if not directory.exists():
50
+ raise PathExistsError(f"Path {str(directory)!r} does not exists.")
51
+
52
+ dps: list[typing.Self] = []
53
+ for root, _, files in os.walk(directory):
54
+ for f in files:
55
+ if f.endswith(".py") and f != "__init__.py":
56
+ module_path = os.path.join(root, f)
57
+ module_name = os.path.splitext(os.path.relpath(module_path, directory))[0]
58
+ module_name = module_name.replace(os.sep, ".")
59
+
60
+ spec = importlib_util.spec_from_file_location(module_name, module_path)
61
+ if spec is None or spec.loader is None:
62
+ continue
63
+
64
+ module = importlib_util.module_from_spec(spec)
65
+ sys.modules[module_name] = module
66
+ spec.loader.exec_module(module)
67
+
68
+ for obj in module.__dict__.values():
69
+ if isinstance(obj, self.__class__):
70
+ dps.append(obj)
71
+
72
+ self.load_many(*dps)
73
+ return bool(dps)
74
+
75
+
76
+ __all__ = ("ABCDispatch",)
@@ -1,98 +1,96 @@
1
- from __future__ import annotations
2
-
3
- import enum
4
- import typing
5
- from reprlib import recursive_repr
6
-
7
- from telegrinder.types.objects import Update
8
-
9
- if typing.TYPE_CHECKING:
10
- from telegrinder.node.composer import NodeCollection
11
-
12
- T = typing.TypeVar("T")
13
-
14
- Key: typing.TypeAlias = str | enum.Enum
15
- AnyValue: typing.TypeAlias = typing.Any
16
-
17
-
18
- class Context(dict[str, AnyValue]):
19
- """Context class like dict & dotdict.
20
-
21
- For example:
22
- ```python
23
- class MyRule(ABCRule[T]):
24
- async def check(self, event: T, ctx: Context) -> bool:
25
- ctx.me = (await event.ctx_api.get_me()).unwrap()
26
- ctx["items"] = [1, 2, 3]
27
- return True
28
- ```
29
- """
30
-
31
- raw_update: Update
32
- node_col: NodeCollection | None = None
33
-
34
- def __init__(self, **kwargs: AnyValue) -> None:
35
- cls_vars = vars(self.__class__)
36
- defaults = {}
37
-
38
- for k in self.__class__.__annotations__:
39
- if k in cls_vars:
40
- defaults[k] = cls_vars[k]
41
- delattr(self.__class__, k)
42
-
43
- dict.__init__(self, **defaults | kwargs)
44
-
45
- @recursive_repr()
46
- def __repr__(self) -> str:
47
- return "{}({})".format(self.__class__.__name__, ", ".join(f"{k}={v!r}" for k, v in self.items()))
48
-
49
- def __setitem__(self, __key: Key, __value: AnyValue) -> None:
50
- dict.__setitem__(self, self.key_to_str(__key), __value)
51
-
52
- def __getitem__(self, __key: Key) -> AnyValue:
53
- return dict.__getitem__(self, self.key_to_str(__key))
54
-
55
- def __delitem__(self, __key: Key) -> None:
56
- dict.__delitem__(self, self.key_to_str(__key))
57
-
58
- def __setattr__(self, __name: str, __value: AnyValue) -> None:
59
- self.__setitem__(__name, __value)
60
-
61
- def __getattr__(self, __name: str) -> AnyValue:
62
- return self.__getitem__(__name)
63
-
64
- def __delattr__(self, __name: str) -> None:
65
- self.__delitem__(__name)
66
-
67
- @staticmethod
68
- def key_to_str(key: Key) -> str:
69
- return key if isinstance(key, str) else str(key.value)
70
-
71
- def copy(self) -> typing.Self:
72
- return self.__class__(**dict.copy(self))
73
-
74
- def set(self, key: Key, value: AnyValue) -> None:
75
- self[key] = value
76
-
77
- @typing.overload
78
- def get(self, key: Key) -> AnyValue | None: ...
79
-
80
- @typing.overload
81
- def get(self, key: Key, default: T) -> T | AnyValue: ...
82
-
83
- @typing.overload
84
- def get(self, key: Key, default: None = None) -> AnyValue | None: ...
85
-
86
- def get(self, key: Key, default: T | None = None) -> T | AnyValue | None:
87
- return dict.get(self, key, default)
88
-
89
- def get_or_set(self, key: Key, default: T) -> T:
90
- if key not in self:
91
- self.set(key, default)
92
- return self.get(key, default)
93
-
94
- def delete(self, key: Key) -> None:
95
- del self[key]
96
-
97
-
98
- __all__ = ("Context",)
1
+ from __future__ import annotations
2
+
3
+ import enum
4
+ import typing
5
+ from reprlib import recursive_repr
6
+
7
+ from telegrinder.types.objects import Update
8
+
9
+ if typing.TYPE_CHECKING:
10
+ from telegrinder.node.composer import NodeCollection
11
+
12
+ type Key = str | enum.Enum
13
+ type AnyValue = typing.Any
14
+
15
+
16
+ class Context(dict[str, AnyValue]):
17
+ """Context class like dict & dotdict.
18
+
19
+ For example:
20
+ ```python
21
+ class MyRule(ABCRule[T]):
22
+ async def check(self, event: T, ctx: Context) -> bool:
23
+ ctx.me = (await event.ctx_api.get_me()).unwrap()
24
+ ctx["items"] = [1, 2, 3]
25
+ return True
26
+ ```
27
+ """
28
+
29
+ raw_update: Update
30
+ node_col: NodeCollection | None = None
31
+
32
+ def __init__(self, **kwargs: AnyValue) -> None:
33
+ cls_vars = vars(self.__class__)
34
+ defaults = {}
35
+
36
+ for k in self.__class__.__annotations__:
37
+ if k in cls_vars:
38
+ defaults[k] = cls_vars[k]
39
+ delattr(self.__class__, k)
40
+
41
+ dict.__init__(self, **defaults | kwargs)
42
+
43
+ @recursive_repr()
44
+ def __repr__(self) -> str:
45
+ return "{}({})".format(self.__class__.__name__, ", ".join(f"{k}={v!r}" for k, v in self.items()))
46
+
47
+ def __setitem__(self, __key: Key, __value: AnyValue) -> None:
48
+ dict.__setitem__(self, self.key_to_str(__key), __value)
49
+
50
+ def __getitem__(self, __key: Key) -> AnyValue:
51
+ return dict.__getitem__(self, self.key_to_str(__key))
52
+
53
+ def __delitem__(self, __key: Key) -> None:
54
+ dict.__delitem__(self, self.key_to_str(__key))
55
+
56
+ def __setattr__(self, __name: str, __value: AnyValue) -> None:
57
+ self.__setitem__(__name, __value)
58
+
59
+ def __getattr__(self, __name: str) -> AnyValue:
60
+ return self.__getitem__(__name)
61
+
62
+ def __delattr__(self, __name: str) -> None:
63
+ self.__delitem__(__name)
64
+
65
+ @staticmethod
66
+ def key_to_str(key: Key) -> str:
67
+ return key if isinstance(key, str) else str(key.value)
68
+
69
+ def copy(self) -> typing.Self:
70
+ return self.__class__(**dict.copy(self))
71
+
72
+ def set(self, key: Key, value: AnyValue) -> None:
73
+ self[key] = value
74
+
75
+ @typing.overload
76
+ def get(self, key: Key) -> AnyValue | None: ...
77
+
78
+ @typing.overload
79
+ def get[T](self, key: Key, default: T) -> T | AnyValue: ...
80
+
81
+ @typing.overload
82
+ def get(self, key: Key, default: None = None) -> AnyValue | None: ...
83
+
84
+ def get[T](self, key: Key, default: T | None = None) -> T | AnyValue | None:
85
+ return dict.get(self, key, default)
86
+
87
+ def get_or_set[T](self, key: Key, default: T) -> T:
88
+ if key not in self:
89
+ self.set(key, default)
90
+ return self.get(key, default)
91
+
92
+ def delete(self, key: Key) -> None:
93
+ del self[key]
94
+
95
+
96
+ __all__ = ("Context",)