mirror of
https://github.com/aiogram/aiogram.git
synced 2025-12-12 18:19:34 +00:00
194 lines
7.1 KiB
Python
194 lines
7.1 KiB
Python
import asyncio
|
|
import io
|
|
import ssl
|
|
from typing import Dict, List, Optional, Union
|
|
|
|
import aiohttp
|
|
import certifi
|
|
|
|
from . import api
|
|
from ..types import ParseMode, base
|
|
from ..utils import json
|
|
from ..utils.auth_widget import check_token
|
|
|
|
|
|
class BaseBot:
|
|
"""
|
|
Base class for bot. It's raw bot.
|
|
"""
|
|
|
|
def __init__(self, token: base.String,
|
|
loop: Optional[Union[asyncio.BaseEventLoop, asyncio.AbstractEventLoop]] = None,
|
|
connections_limit: Optional[base.Integer] = None,
|
|
proxy: Optional[base.String] = None, proxy_auth: Optional[aiohttp.BasicAuth] = None,
|
|
validate_token: Optional[base.Boolean] = True,
|
|
parse_mode=None):
|
|
"""
|
|
Instructions how to get Bot token is found here: https://core.telegram.org/bots#3-how-do-i-create-a-bot
|
|
|
|
:param token: token from @BotFather
|
|
:type token: :obj:`str`
|
|
:param loop: event loop
|
|
:type loop: Optional Union :obj:`asyncio.BaseEventLoop`, :obj:`asyncio.AbstractEventLoop`
|
|
:param connections_limit: connections limit for aiohttp.ClientSession
|
|
:type connections_limit: :obj:`int`
|
|
:param proxy: HTTP proxy URL
|
|
:type proxy: :obj:`str`
|
|
:param proxy_auth: Authentication information
|
|
:type proxy_auth: Optional :obj:`aiohttp.BasicAuth`
|
|
:param validate_token: Validate token.
|
|
:type validate_token: :obj:`bool`
|
|
:param parse_mode: You can set default parse mode
|
|
:type parse_mode: :obj:`str`
|
|
:raise: when token is invalid throw an :obj:`aiogram.utils.exceptions.ValidationError`
|
|
"""
|
|
# Authentication
|
|
if validate_token:
|
|
api.check_token(token)
|
|
self.__token = token
|
|
|
|
self.proxy = proxy
|
|
self.proxy_auth = proxy_auth
|
|
|
|
# Asyncio loop instance
|
|
if loop is None:
|
|
loop = asyncio.get_event_loop()
|
|
self.loop = loop
|
|
|
|
# aiohttp main session
|
|
ssl_context = ssl.create_default_context(cafile=certifi.where())
|
|
|
|
if isinstance(proxy, str) and (proxy.startswith('socks5://') or proxy.startswith('socks4://')):
|
|
from aiohttp_socks import SocksConnector
|
|
from aiohttp_socks.helpers import parse_socks_url
|
|
|
|
socks_ver, host, port, username, password = parse_socks_url(proxy)
|
|
if proxy_auth:
|
|
if not username:
|
|
username = proxy_auth.login
|
|
if not password:
|
|
password = proxy_auth.password
|
|
|
|
connector = SocksConnector(socks_ver=socks_ver, host=host, port=port,
|
|
username=username, password=password,
|
|
limit=connections_limit, ssl_context=ssl_context,
|
|
loop=self.loop)
|
|
|
|
self.proxy = None
|
|
self.proxy_auth = None
|
|
else:
|
|
connector = aiohttp.TCPConnector(limit=connections_limit, ssl_context=ssl_context,
|
|
loop=self.loop)
|
|
|
|
self.session = aiohttp.ClientSession(connector=connector, loop=self.loop, json_serialize=json.dumps)
|
|
|
|
self.parse_mode = parse_mode
|
|
|
|
async def close(self):
|
|
"""
|
|
Close all client sessions
|
|
"""
|
|
await self.session.close()
|
|
|
|
async def request(self, method: base.String,
|
|
data: Optional[Dict] = None,
|
|
files: Optional[Dict] = None, **kwargs) -> Union[List, Dict, base.Boolean]:
|
|
"""
|
|
Make an request to Telegram Bot API
|
|
|
|
https://core.telegram.org/bots/api#making-requests
|
|
|
|
:param method: API method
|
|
:type method: :obj:`str`
|
|
:param data: request parameters
|
|
:type data: :obj:`dict`
|
|
:param files: files
|
|
:type files: :obj:`dict`
|
|
:return: result
|
|
:rtype: Union[List, Dict]
|
|
:raise: :obj:`aiogram.exceptions.TelegramApiError`
|
|
"""
|
|
return await api.make_request(self.session, self.__token, method, data, files,
|
|
proxy=self.proxy, proxy_auth=self.proxy_auth, **kwargs)
|
|
|
|
async def download_file(self, file_path: base.String,
|
|
destination: Optional[base.InputFile] = None,
|
|
timeout: Optional[base.Integer] = 30,
|
|
chunk_size: Optional[base.Integer] = 65536,
|
|
seek: Optional[base.Boolean] = True) -> Union[io.BytesIO, io.FileIO]:
|
|
"""
|
|
Download file by file_path to destination
|
|
|
|
if You want to automatically create destination (:class:`io.BytesIO`) use default
|
|
value of destination and handle result of this method.
|
|
|
|
:param file_path: file path on telegram server (You can get it from :obj:`aiogram.types.File`)
|
|
:type file_path: :obj:`str`
|
|
:param destination: filename or instance of :class:`io.IOBase`. For e. g. :class:`io.BytesIO`
|
|
:param timeout: Integer
|
|
:param chunk_size: Integer
|
|
:param seek: Boolean - go to start of file when downloading is finished.
|
|
:return: destination
|
|
"""
|
|
if destination is None:
|
|
destination = io.BytesIO()
|
|
|
|
url = api.Methods.file_url(token=self.__token, path=file_path)
|
|
|
|
dest = destination if isinstance(destination, io.IOBase) else open(destination, 'wb')
|
|
async with self.session.get(url, timeout=timeout, proxy=self.proxy, proxy_auth=self.proxy_auth) as response:
|
|
while True:
|
|
chunk = await response.content.read(chunk_size)
|
|
if not chunk:
|
|
break
|
|
dest.write(chunk)
|
|
dest.flush()
|
|
if seek:
|
|
dest.seek(0)
|
|
return dest
|
|
|
|
async def send_file(self, file_type, method, file, payload) -> Union[Dict, base.Boolean]:
|
|
"""
|
|
Send file
|
|
|
|
https://core.telegram.org/bots/api#inputfile
|
|
|
|
:param file_type: field name
|
|
:param method: API method
|
|
:param file: String or io.IOBase
|
|
:param payload: request payload
|
|
:return: response
|
|
"""
|
|
if file is None:
|
|
files = {}
|
|
elif isinstance(file, str):
|
|
# You can use file ID or URL in the most of requests
|
|
payload[file_type] = file
|
|
files = None
|
|
else:
|
|
files = {file_type: file}
|
|
|
|
return await self.request(method, payload, files)
|
|
|
|
@property
|
|
def parse_mode(self):
|
|
return getattr(self, '_parse_mode', None)
|
|
|
|
@parse_mode.setter
|
|
def parse_mode(self, value):
|
|
if value is None:
|
|
setattr(self, '_parse_mode', None)
|
|
else:
|
|
if not isinstance(value, str):
|
|
raise TypeError(f"Parse mode must be str, not {type(value)}")
|
|
value = value.lower()
|
|
if value not in ParseMode.all():
|
|
raise ValueError(f"Parse mode must be one of {ParseMode.all()}")
|
|
setattr(self, '_parse_mode', value)
|
|
|
|
@parse_mode.deleter
|
|
def parse_mode(self):
|
|
self.parse_mode = None
|
|
|
|
def check_auth_widget(self, data):
|
|
return check_token(data, self.__token)
|