- Asyncio and aiohttp based
- All Telegram Bot API types and methods supported
- Tracks Telegram Bot API 7.2 (released March 31, 2024)
- Bot API rate limit support
- Both long polling and webhooks supported
- Fully type annotated (PEP 484)
aiotgbot is available on PyPI. Use pip to install it:
pip install aiotgbot- Python 3.11–3.14
- aiohttp
- aiojobs
- msgspec
- tenacity
- frozenlist
- aiofreqlimit
- yarl
from typing import AsyncIterator
from aiotgbot import (
Bot,
BotUpdate,
BotUpdateKey,
HandlerTable,
PollBot,
PrivateChatFilter,
Runner,
)
from aiotgbot.storage_memory import MemoryStorage
handlers = HandlerTable()
@handlers.message(filters=[PrivateChatFilter()])
async def reply_private_message(bot: Bot, update: BotUpdate) -> None:
assert update.message is not None
name = (
f"{update.message.chat.first_name} "
f"{update.message.chat.last_name}"
)
update["greeting_count"] = update.get("greeting_count", 0) + 1
await bot.send_message(update.message.chat.id, f"Hello, {name}!")
async def run_context(runner: Runner) -> AsyncIterator[None]:
storage = MemoryStorage()
await storage.connect()
handlers.freeze()
bot = PollBot(runner["token"], handlers, storage)
await bot.start()
yield
await bot.stop()
await storage.close()
def main() -> None:
runner = Runner(run_context)
runner["token"] = "some:token"
runner.run()
if __name__ == "__main__":
main()New features:
- BotUpdateKey – a dedicated typed key object for items stored on
BotUpdate. BotUpdateexposesget_typed(key),set_typed(key, value), anddel_typed(key)helpers for working withBotUpdateKeyinstances.- Each
BotUpdateKeyenforces runtime type checking viaisinstance()so handlers only see the expected payload.
BotUpdate remains a regular mutable mapping so filters and handlers can stash arbitrary helper objects between each other. To keep data structured, use BotUpdateKey which enforces types per slot:
from dataclasses import dataclass
from aiotgbot import BotUpdateKey
@dataclass
class Session:
trace_id: str
retries: int
session_key = BotUpdateKey("session", Session)
async def my_handler(bot: Bot, update: BotUpdate) -> None:
if session_key.name not in update:
update.set_typed(session_key, Session(trace_id="abc", retries=0))
session = update.get_typed(session_key)
...We use Prek as a drop-in pre-commit replacement backed by uv so hook environments resolve quickly and reproducibly. Install it once and run the configured Ruff, mypy --strict, and Basedpyright checks via:
uv tool install prek
prek install
prek run --all-filesprek run reads .pre-commit-config.yaml, so you can still target a subset of hooks or files during local development.
mise.toml at the repo root mirrors the common workflows, so you can rely on mise instead of remembering the raw commands. Trust the config once via mise trust mise.toml and then run, for example:
mise run lint
mise run mypy
mise run basedpyright
mise run test