mirror of
https://github.com/aiogram/aiogram.git
synced 2026-04-08 16:37:47 +00:00
Base implementation
This commit is contained in:
parent
2c756a1689
commit
0956450cd9
6 changed files with 128 additions and 28 deletions
|
|
@ -4,6 +4,10 @@ from typing import Any, Awaitable, Callable, Dict, Iterator, Optional, Tuple
|
|||
from aiogram.dispatcher.middlewares.base import BaseMiddleware
|
||||
from aiogram.types import Chat, TelegramObject, Update, User
|
||||
|
||||
EVENT_FROM_USER_KEY = "event_from_user"
|
||||
EVENT_CHAT_KEY = "event_chat"
|
||||
EVENT_THREAD_ID_KEY = "event_thread_id"
|
||||
|
||||
|
||||
class UserContextMiddleware(BaseMiddleware):
|
||||
async def __call__(
|
||||
|
|
@ -14,12 +18,14 @@ class UserContextMiddleware(BaseMiddleware):
|
|||
) -> Any:
|
||||
if not isinstance(event, Update):
|
||||
raise RuntimeError("UserContextMiddleware got an unexpected event type!")
|
||||
chat, user = self.resolve_event_context(event=event)
|
||||
chat, user, thread_id = self.resolve_event_context(event=event)
|
||||
with self.context(chat=chat, user=user):
|
||||
if user is not None:
|
||||
data["event_from_user"] = user
|
||||
data[EVENT_FROM_USER_KEY] = user
|
||||
if chat is not None:
|
||||
data["event_chat"] = chat
|
||||
data[EVENT_CHAT_KEY] = chat
|
||||
if thread_id is not None:
|
||||
data[EVENT_THREAD_ID_KEY] = thread_id
|
||||
return await handler(event, data)
|
||||
|
||||
@contextmanager
|
||||
|
|
@ -39,36 +45,54 @@ class UserContextMiddleware(BaseMiddleware):
|
|||
user.reset_current(user_token)
|
||||
|
||||
@classmethod
|
||||
def resolve_event_context(cls, event: Update) -> Tuple[Optional[Chat], Optional[User]]:
|
||||
def resolve_event_context(
|
||||
cls, event: Update
|
||||
) -> Tuple[Optional[Chat], Optional[User], Optional[int]]:
|
||||
"""
|
||||
Resolve chat and user instance from Update object
|
||||
"""
|
||||
if event.message:
|
||||
return event.message.chat, event.message.from_user
|
||||
return (
|
||||
event.message.chat,
|
||||
event.message.from_user,
|
||||
event.message.message_thread_id if event.message.is_topic_message else None,
|
||||
)
|
||||
if event.edited_message:
|
||||
return event.edited_message.chat, event.edited_message.from_user
|
||||
return (
|
||||
event.edited_message.chat,
|
||||
event.edited_message.from_user,
|
||||
event.edited_message.message_thread_id
|
||||
if event.edited_message.is_topic_message
|
||||
else None,
|
||||
)
|
||||
if event.channel_post:
|
||||
return event.channel_post.chat, None
|
||||
return event.channel_post.chat, None, None
|
||||
if event.edited_channel_post:
|
||||
return event.edited_channel_post.chat, None
|
||||
return event.edited_channel_post.chat, None, None
|
||||
if event.inline_query:
|
||||
return None, event.inline_query.from_user
|
||||
return None, event.inline_query.from_user, None
|
||||
if event.chosen_inline_result:
|
||||
return None, event.chosen_inline_result.from_user
|
||||
return None, event.chosen_inline_result.from_user, None
|
||||
if event.callback_query:
|
||||
if event.callback_query.message:
|
||||
return event.callback_query.message.chat, event.callback_query.from_user
|
||||
return None, event.callback_query.from_user
|
||||
return (
|
||||
event.callback_query.message.chat,
|
||||
event.callback_query.from_user,
|
||||
event.callback_query.message.message_thread_id
|
||||
if event.callback_query.message.is_topic_message
|
||||
else None,
|
||||
)
|
||||
return None, event.callback_query.from_user, None
|
||||
if event.shipping_query:
|
||||
return None, event.shipping_query.from_user
|
||||
return None, event.shipping_query.from_user, None
|
||||
if event.pre_checkout_query:
|
||||
return None, event.pre_checkout_query.from_user
|
||||
return None, event.pre_checkout_query.from_user, None
|
||||
if event.poll_answer:
|
||||
return None, event.poll_answer.user
|
||||
return None, event.poll_answer.user, None
|
||||
if event.my_chat_member:
|
||||
return event.my_chat_member.chat, event.my_chat_member.from_user
|
||||
return event.my_chat_member.chat, event.my_chat_member.from_user, None
|
||||
if event.chat_member:
|
||||
return event.chat_member.chat, event.chat_member.from_user
|
||||
return event.chat_member.chat, event.chat_member.from_user, None
|
||||
if event.chat_join_request:
|
||||
return event.chat_join_request.chat, event.chat_join_request.from_user
|
||||
return None, None
|
||||
return event.chat_join_request.chat, event.chat_join_request.from_user, None
|
||||
return None, None, None
|
||||
|
|
|
|||
|
|
@ -47,25 +47,42 @@ class FSMContextMiddleware(BaseMiddleware):
|
|||
) -> Optional[FSMContext]:
|
||||
user = data.get("event_from_user")
|
||||
chat = data.get("event_chat")
|
||||
thread_id = data.get("event_thread_id")
|
||||
chat_id = chat.id if chat else None
|
||||
user_id = user.id if user else None
|
||||
return self.resolve_context(bot=bot, chat_id=chat_id, user_id=user_id, destiny=destiny)
|
||||
return self.resolve_context(
|
||||
bot=bot,
|
||||
chat_id=chat_id,
|
||||
user_id=user_id,
|
||||
thread_id=thread_id,
|
||||
destiny=destiny,
|
||||
)
|
||||
|
||||
def resolve_context(
|
||||
self,
|
||||
bot: Bot,
|
||||
chat_id: Optional[int],
|
||||
user_id: Optional[int],
|
||||
thread_id: Optional[int] = None,
|
||||
destiny: str = DEFAULT_DESTINY,
|
||||
) -> Optional[FSMContext]:
|
||||
if chat_id is None:
|
||||
chat_id = user_id
|
||||
|
||||
if chat_id is not None and user_id is not None:
|
||||
chat_id, user_id = apply_strategy(
|
||||
chat_id=chat_id, user_id=user_id, strategy=self.strategy
|
||||
chat_id, user_id, thread_id = apply_strategy(
|
||||
chat_id=chat_id,
|
||||
user_id=user_id,
|
||||
thread_id=thread_id,
|
||||
strategy=self.strategy,
|
||||
)
|
||||
return self.get_context(
|
||||
bot=bot,
|
||||
chat_id=chat_id,
|
||||
user_id=user_id,
|
||||
thread_id=thread_id,
|
||||
destiny=destiny,
|
||||
)
|
||||
return self.get_context(bot=bot, chat_id=chat_id, user_id=user_id, destiny=destiny)
|
||||
return None
|
||||
|
||||
def get_context(
|
||||
|
|
@ -73,6 +90,7 @@ class FSMContextMiddleware(BaseMiddleware):
|
|||
bot: Bot,
|
||||
chat_id: int,
|
||||
user_id: int,
|
||||
thread_id: Optional[int] = None,
|
||||
destiny: str = DEFAULT_DESTINY,
|
||||
) -> FSMContext:
|
||||
return FSMContext(
|
||||
|
|
@ -81,6 +99,7 @@ class FSMContextMiddleware(BaseMiddleware):
|
|||
user_id=user_id,
|
||||
chat_id=chat_id,
|
||||
bot_id=bot.id,
|
||||
thread_id=thread_id,
|
||||
destiny=destiny,
|
||||
),
|
||||
)
|
||||
|
|
|
|||
|
|
@ -15,6 +15,7 @@ class StorageKey:
|
|||
bot_id: int
|
||||
chat_id: int
|
||||
user_id: int
|
||||
thread_id: Optional[int] = None
|
||||
destiny: str = DEFAULT_DESTINY
|
||||
|
||||
|
||||
|
|
|
|||
|
|
@ -71,6 +71,8 @@ class DefaultKeyBuilder(KeyBuilder):
|
|||
if self.with_bot_id:
|
||||
parts.append(str(key.bot_id))
|
||||
parts.extend([str(key.chat_id), str(key.user_id)])
|
||||
if key.thread_id:
|
||||
parts.append(str(key.thread_id))
|
||||
if self.with_destiny:
|
||||
parts.append(key.destiny)
|
||||
elif key.destiny != DEFAULT_DESTINY:
|
||||
|
|
|
|||
|
|
@ -1,16 +1,24 @@
|
|||
from enum import Enum, auto
|
||||
from typing import Tuple
|
||||
from typing import Optional, Tuple
|
||||
|
||||
|
||||
class FSMStrategy(Enum):
|
||||
USER_IN_CHAT = auto()
|
||||
CHAT = auto()
|
||||
GLOBAL_USER = auto()
|
||||
USER_IN_THREAD = auto()
|
||||
|
||||
|
||||
def apply_strategy(chat_id: int, user_id: int, strategy: FSMStrategy) -> Tuple[int, int]:
|
||||
def apply_strategy(
|
||||
strategy: FSMStrategy,
|
||||
chat_id: int,
|
||||
user_id: int,
|
||||
thread_id: int | None = None,
|
||||
) -> Tuple[int, int, Optional[int]]:
|
||||
if strategy == FSMStrategy.CHAT:
|
||||
return chat_id, chat_id
|
||||
return chat_id, chat_id, None
|
||||
if strategy == FSMStrategy.GLOBAL_USER:
|
||||
return user_id, user_id
|
||||
return chat_id, user_id
|
||||
return user_id, user_id, None
|
||||
if strategy == FSMStrategy.USER_IN_THREAD:
|
||||
return user_id, chat_id, thread_id
|
||||
return chat_id, user_id, None
|
||||
|
|
|
|||
|
|
@ -26,3 +26,49 @@ def create_tg_link(link: str, **kwargs: Any) -> str:
|
|||
|
||||
def create_telegram_link(*path: str, **kwargs: Any) -> str:
|
||||
return _format_url("https://t.me", *path, **kwargs)
|
||||
|
||||
|
||||
def create_channel_bot_link(
|
||||
username: str,
|
||||
parameter: Optional[str] = None,
|
||||
change_info: bool = False,
|
||||
post_messages: bool = False,
|
||||
edit_messages: bool = False,
|
||||
delete_messages: bool = False,
|
||||
restrict_members: bool = False,
|
||||
invite_users: bool = False,
|
||||
pin_messages: bool = False,
|
||||
promote_members: bool = False,
|
||||
manage_video_chats: bool = False,
|
||||
anonymous: bool = False,
|
||||
manage_chat: bool = False,
|
||||
) -> str:
|
||||
params = {}
|
||||
if parameter is not None:
|
||||
params["startgroup"] = parameter
|
||||
permissions = []
|
||||
if change_info:
|
||||
permissions.append("change_info")
|
||||
if post_messages:
|
||||
permissions.append("post_messages")
|
||||
if edit_messages:
|
||||
permissions.append("edit_messages")
|
||||
if delete_messages:
|
||||
permissions.append("delete_messages")
|
||||
if restrict_members:
|
||||
permissions.append("restrict_members")
|
||||
if invite_users:
|
||||
permissions.append("invite_users")
|
||||
if pin_messages:
|
||||
permissions.append("pin_messages")
|
||||
if promote_members:
|
||||
permissions.append("promote_members")
|
||||
if manage_video_chats:
|
||||
permissions.append("manage_video_chats")
|
||||
if anonymous:
|
||||
permissions.append("anonymous")
|
||||
if manage_chat:
|
||||
permissions.append("manage_chat")
|
||||
if permissions:
|
||||
params["admin"] = "+".join(permissions)
|
||||
return create_telegram_link(username, **params)
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue