mirror of
https://github.com/aiogram/aiogram.git
synced 2025-12-06 07:50:32 +00:00
Some checks failed
Tests / tests (macos-latest, 3.10) (push) Has been cancelled
Tests / tests (macos-latest, 3.11) (push) Has been cancelled
Tests / tests (macos-latest, 3.12) (push) Has been cancelled
Tests / tests (macos-latest, 3.13) (push) Has been cancelled
Tests / tests (ubuntu-latest, 3.10) (push) Has been cancelled
Tests / tests (ubuntu-latest, 3.11) (push) Has been cancelled
Tests / tests (ubuntu-latest, 3.12) (push) Has been cancelled
Tests / tests (ubuntu-latest, 3.13) (push) Has been cancelled
Tests / tests (windows-latest, 3.10) (push) Has been cancelled
Tests / tests (windows-latest, 3.11) (push) Has been cancelled
Tests / tests (windows-latest, 3.12) (push) Has been cancelled
Tests / tests (windows-latest, 3.13) (push) Has been cancelled
Tests / pypy-tests (macos-latest, pypy3.10) (push) Has been cancelled
Tests / pypy-tests (macos-latest, pypy3.11) (push) Has been cancelled
Tests / pypy-tests (ubuntu-latest, pypy3.10) (push) Has been cancelled
Tests / pypy-tests (ubuntu-latest, pypy3.11) (push) Has been cancelled
* Drop py3.9 and pypy3.9 Add pypy3.11 (testing) into `tests.yml` Remove py3.9 from matrix in `tests.yml` Refactor not auto-gen code to be compatible with py3.10+, droping ugly 3.9 annotation. Replace some `from typing` imports to `from collections.abc`, due to deprecation Add `from __future__ import annotations` and `if TYPE_CHECKING:` where possible Add some `noqa` to calm down Ruff in some places, if Ruff will be used as default linting+formatting tool in future Replace some relative imports to absolute Sort `__all__` tuples in `__init__.py` and some other `.py` files Sort `__slots__` tuples in classes Split raises into `msg` and `raise` (`EM101`, `EM102`) to not duplicate error message in the traceback Add `Self` from `typing_extenstion` where possible Resolve typing problem in `aiogram/filters/command.py:18` Concatenate nested `if` statements Convert `HandlerContainer` into a dataclass in `aiogram/fsm/scene.py` Bump tests docker-compose.yml `redis:6-alpine` -> `redis:8-alpine` Bump tests docker-compose.yml `mongo:7.0.6` -> `mongo:8.0.14` Bump pre-commit-config `black==24.4.2` -> `black==25.9.0` Bump pre-commit-config `ruff==0.5.1` -> `ruff==0.13.3` Update Makefile lint for ruff to show fixes Add `make outdated` into Makefile Use `pathlib` instead of `os.path` Bump `redis[hiredis]>=5.0.1,<5.3.0` -> `redis[hiredis]>=6.2.0,<7` Bump `cryptography>=43.0.0` -> `cryptography>=46.0.0` due to security reasons Bump `pytz~=2023.3` -> `pytz~=2025.2` Bump `pycryptodomex~=3.19.0` -> `pycryptodomex~=3.23.0` due to security reasons Bump linting and formatting tools * Add `1726.removal.rst` * Update aiogram/utils/dataclass.py Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update aiogram/filters/callback_data.py Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update 1726.removal.rst * Remove `outdated` from Makefile * Add `__slots__` to `HandlerContainer` * Remove unused imports * Add `@dataclass` with `slots=True` to `HandlerContainer` --------- Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
97 lines
2.7 KiB
Python
97 lines
2.7 KiB
Python
import asyncio
|
|
import logging
|
|
import sys
|
|
from os import getenv
|
|
|
|
from aiogram import Bot, Dispatcher, Router
|
|
from aiogram.client.default import DefaultBotProperties
|
|
from aiogram.enums import ParseMode
|
|
from aiogram.filters import LEAVE_TRANSITION, ChatMemberUpdatedFilter, CommandStart
|
|
from aiogram.types import (
|
|
CallbackQuery,
|
|
ChatMemberUpdated,
|
|
InlineKeyboardButton,
|
|
InlineKeyboardMarkup,
|
|
Message,
|
|
)
|
|
from aiogram.utils.markdown import hbold, hcode
|
|
|
|
TOKEN = getenv("BOT_TOKEN")
|
|
|
|
logger = logging.getLogger(__name__)
|
|
logging.basicConfig(level=logging.INFO)
|
|
|
|
router = Router()
|
|
|
|
|
|
@router.message(CommandStart())
|
|
async def command_start_handler(message: Message) -> None:
|
|
"""
|
|
This handler receives messages with `/start` command
|
|
"""
|
|
|
|
await message.answer(
|
|
f"Hello, {hbold(message.from_user.full_name)}!",
|
|
reply_markup=InlineKeyboardMarkup(
|
|
inline_keyboard=[[InlineKeyboardButton(text="Tap me, bro", callback_data="*")]],
|
|
),
|
|
)
|
|
|
|
|
|
@router.chat_member()
|
|
async def chat_member_update(chat_member: ChatMemberUpdated, bot: Bot) -> None:
|
|
await bot.send_message(
|
|
chat_member.chat.id,
|
|
f"Member {hcode(chat_member.from_user.id)} was changed "
|
|
f"from {chat_member.old_chat_member.status} to {chat_member.new_chat_member.status}",
|
|
)
|
|
|
|
|
|
# this router will use only callback_query updates
|
|
sub_router = Router()
|
|
|
|
|
|
@sub_router.callback_query()
|
|
async def callback_tap_me(callback_query: CallbackQuery) -> None:
|
|
await callback_query.answer("Yeah good, now I'm fine")
|
|
|
|
|
|
# this router will use only edited_message updates
|
|
sub_sub_router = Router()
|
|
|
|
|
|
@sub_sub_router.edited_message()
|
|
async def edited_message_handler(edited_message: Message) -> None:
|
|
await edited_message.reply("Message was edited, Big Brother watches you")
|
|
|
|
|
|
# this router will use only my_chat_member updates
|
|
deep_dark_router = Router()
|
|
|
|
|
|
@deep_dark_router.my_chat_member(~ChatMemberUpdatedFilter(~LEAVE_TRANSITION))
|
|
async def my_chat_member_change(chat_member: ChatMemberUpdated, bot: Bot) -> None:
|
|
await bot.send_message(
|
|
chat_member.chat.id,
|
|
f"This Bot`s status was changed from {hbold(chat_member.old_chat_member.status)} "
|
|
f"to {hbold(chat_member.new_chat_member.status)}",
|
|
)
|
|
|
|
|
|
async def main() -> None:
|
|
# Initialize Bot instance with default bot properties which will be passed to all API calls
|
|
bot = Bot(token=TOKEN, default=DefaultBotProperties(parse_mode=ParseMode.HTML))
|
|
|
|
dp = Dispatcher()
|
|
|
|
sub_router.include_router(deep_dark_router)
|
|
router.include_routers(sub_router, sub_sub_router)
|
|
dp.include_router(router)
|
|
|
|
# Start event dispatching
|
|
await dp.start_polling(bot)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
logging.basicConfig(level=logging.INFO, stream=sys.stdout)
|
|
asyncio.run(main())
|