2022-03-28 16:39:36 +02:00
|
|
|
import asyncio
|
2021-04-09 13:31:35 +02:00
|
|
|
import datetime
|
|
|
|
import logging
|
2022-03-28 16:39:36 +02:00
|
|
|
import os.path
|
2021-04-24 14:19:24 +02:00
|
|
|
from typing import (
|
|
|
|
Optional,
|
|
|
|
Union,
|
|
|
|
)
|
2021-04-09 13:31:35 +02:00
|
|
|
|
|
|
|
from aiokit import AioThing
|
2022-03-28 16:39:36 +02:00
|
|
|
from izihawa_utils.random import generate_request_id
|
2021-04-09 13:31:35 +02:00
|
|
|
from library.logging import error_log
|
|
|
|
from telethon import (
|
|
|
|
TelegramClient,
|
|
|
|
connection,
|
|
|
|
sessions,
|
|
|
|
)
|
|
|
|
from tenacity import ( # noqa
|
|
|
|
retry,
|
|
|
|
retry_if_exception_type,
|
2022-03-28 16:39:36 +02:00
|
|
|
stop_after_attempt,
|
2021-04-09 13:31:35 +02:00
|
|
|
wait_fixed,
|
|
|
|
)
|
|
|
|
|
|
|
|
from .session_backend import AlchemySessionContainer
|
|
|
|
|
|
|
|
|
|
|
|
class BaseTelegramClient(AioThing):
|
2021-04-24 14:19:24 +02:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
app_id: Union[int, str],
|
|
|
|
app_hash: str,
|
|
|
|
database: dict,
|
2022-03-28 16:39:36 +02:00
|
|
|
phone: Optional[str] = None,
|
|
|
|
password: Optional[str] = None,
|
2021-04-24 14:19:24 +02:00
|
|
|
bot_token: Optional[str] = None,
|
|
|
|
mtproxy: Optional[dict] = None,
|
|
|
|
flood_sleep_threshold: int = 60,
|
|
|
|
):
|
|
|
|
super().__init__()
|
|
|
|
if not app_id or not app_hash:
|
|
|
|
raise ValueError(
|
|
|
|
'Your API ID or Hash cannot be empty or None. Set up telegram.app_id and/or telegram.app_hash'
|
|
|
|
)
|
2021-04-09 13:31:35 +02:00
|
|
|
self._telegram_client = TelegramClient(
|
|
|
|
self._get_session(database),
|
|
|
|
app_id,
|
|
|
|
app_hash,
|
|
|
|
flood_sleep_threshold=flood_sleep_threshold,
|
|
|
|
**self._get_proxy(mtproxy=mtproxy),
|
|
|
|
)
|
2022-03-28 16:39:36 +02:00
|
|
|
self.phone = phone
|
|
|
|
self.password = password
|
2021-04-09 13:31:35 +02:00
|
|
|
self.bot_token = bot_token
|
|
|
|
|
|
|
|
def _get_session(self, database):
|
|
|
|
if database.get('drivername') == 'postgresql':
|
|
|
|
self.container = AlchemySessionContainer(
|
|
|
|
f"{database['drivername']}://"
|
|
|
|
f"{database['username']}:"
|
|
|
|
f"{database['password']}@"
|
|
|
|
f"{database['host']}:"
|
|
|
|
f"{database['port']}/"
|
|
|
|
f"{database['database']}",
|
|
|
|
session=False,
|
|
|
|
manage_tables=False,
|
|
|
|
)
|
|
|
|
return self.container.new_session(database['session_id'])
|
|
|
|
else:
|
|
|
|
return sessions.SQLiteSession(session_id=database['session_id'])
|
|
|
|
|
|
|
|
def _get_proxy(self, mtproxy=None):
|
|
|
|
if mtproxy and mtproxy.get('enabled', True):
|
|
|
|
proxy_config = mtproxy
|
|
|
|
return {
|
|
|
|
'connection': connection.tcpmtproxy.ConnectionTcpMTProxyRandomizedIntermediate,
|
|
|
|
'proxy': (proxy_config['url'], proxy_config['port'], proxy_config['secret'])
|
|
|
|
}
|
|
|
|
return {}
|
|
|
|
|
2022-03-28 16:39:36 +02:00
|
|
|
@retry(retry=retry_if_exception_type(ConnectionError), stop=stop_after_attempt(3), wait=wait_fixed(5))
|
2021-04-09 13:31:35 +02:00
|
|
|
async def start(self):
|
2022-03-28 16:39:36 +02:00
|
|
|
logging.getLogger('debug').info({'mode': 'telegram', 'action': 'starting'})
|
|
|
|
await self._telegram_client.start(
|
|
|
|
phone=lambda: self.phone,
|
|
|
|
bot_token=self.bot_token,
|
|
|
|
password=self.password,
|
|
|
|
code_callback=self.polling_file,
|
|
|
|
)
|
|
|
|
logging.getLogger('debug').info({'mode': 'telegram', 'action': 'started'})
|
|
|
|
|
|
|
|
async def polling_file(self):
|
|
|
|
fname = '/tmp/telegram_code'
|
|
|
|
while not os.path.exists(fname):
|
|
|
|
await asyncio.sleep(5.0)
|
|
|
|
with open(fname, 'r') as code_file:
|
|
|
|
return code_file.read().strip()
|
2021-04-09 13:31:35 +02:00
|
|
|
|
|
|
|
async def stop(self):
|
|
|
|
return await self.disconnect()
|
|
|
|
|
|
|
|
def add_event_handler(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.add_event_handler(*args, **kwargs)
|
|
|
|
|
|
|
|
def catch_up(self):
|
|
|
|
return self._telegram_client.catch_up()
|
|
|
|
|
|
|
|
def delete_messages(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.delete_messages(*args, **kwargs)
|
|
|
|
|
|
|
|
def disconnect(self):
|
|
|
|
return self._telegram_client.disconnect()
|
|
|
|
|
|
|
|
@property
|
|
|
|
def disconnected(self):
|
|
|
|
return self._telegram_client.disconnected
|
|
|
|
|
|
|
|
def download_document(self, *args, **kwargs):
|
|
|
|
return self._telegram_client._download_document(
|
|
|
|
*args,
|
|
|
|
date=datetime.datetime.now(),
|
|
|
|
thumb=None,
|
|
|
|
progress_callback=None,
|
|
|
|
msg_data=None,
|
|
|
|
**kwargs,
|
|
|
|
)
|
|
|
|
|
|
|
|
def edit_message(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.edit_message(*args, **kwargs)
|
|
|
|
|
|
|
|
def edit_permissions(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.edit_permissions(*args, **kwargs)
|
|
|
|
|
|
|
|
def forward_messages(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.forward_messages(*args, **kwargs)
|
|
|
|
|
|
|
|
def get_entity(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.get_entity(*args, **kwargs)
|
|
|
|
|
|
|
|
def get_input_entity(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.get_input_entity(*args, **kwargs)
|
|
|
|
|
2022-03-28 16:39:36 +02:00
|
|
|
def get_permissions(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.get_permissions(*args, **kwargs)
|
|
|
|
|
2021-04-09 13:31:35 +02:00
|
|
|
def iter_admin_log(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.iter_admin_log(*args, **kwargs)
|
|
|
|
|
|
|
|
def iter_messages(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.iter_messages(*args, **kwargs)
|
|
|
|
|
|
|
|
def list_event_handlers(self):
|
|
|
|
return self._telegram_client.list_event_handlers()
|
|
|
|
|
|
|
|
def remove_event_handlers(self):
|
|
|
|
for handler in reversed(self.list_event_handlers()):
|
|
|
|
self._telegram_client.remove_event_handler(*handler)
|
|
|
|
|
|
|
|
def run_until_disconnected(self):
|
|
|
|
return self._telegram_client.run_until_disconnected()
|
|
|
|
|
|
|
|
def send_message(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.send_message(*args, **kwargs)
|
|
|
|
|
|
|
|
def send_file(self, *args, **kwargs):
|
|
|
|
return self._telegram_client.send_file(*args, **kwargs)
|
|
|
|
|
|
|
|
def __call__(self, *args, **kwargs):
|
|
|
|
return self._telegram_client(*args, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
class RequestContext:
|
|
|
|
def __init__(self, bot_name, chat, request_id: str = None, request_id_length: int = 12):
|
|
|
|
self.bot_name = bot_name
|
|
|
|
self.chat = chat
|
|
|
|
self.request_id = request_id or RequestContext.generate_request_id(request_id_length)
|
|
|
|
self.default_fields = {
|
|
|
|
'bot_name': self.bot_name,
|
2021-04-12 19:47:21 +02:00
|
|
|
'chat_id': self.chat.chat_id,
|
2021-04-09 13:31:35 +02:00
|
|
|
'request_id': self.request_id,
|
|
|
|
}
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def generate_request_id(length):
|
2022-03-28 16:39:36 +02:00
|
|
|
return generate_request_id(length)
|
2021-04-09 13:31:35 +02:00
|
|
|
|
|
|
|
def add_default_fields(self, **fields):
|
|
|
|
self.default_fields.update(fields)
|
|
|
|
|
|
|
|
def statbox(self, **kwargs):
|
|
|
|
logging.getLogger('statbox').info(
|
|
|
|
msg=dict(
|
|
|
|
**self.default_fields,
|
|
|
|
**kwargs,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
def error_log(self, e, level=logging.ERROR, **fields):
|
|
|
|
all_fields = {**self.default_fields, **fields}
|
|
|
|
error_log(e, level=level, **all_fields)
|