outside-fetcher/tasks/channel_history.py

127 lines
3.4 KiB
Python
Raw Normal View History

2019-11-03 22:11:45 +01:00
import pyrogram
import aiohttp
2019-11-04 15:08:33 +01:00
import json
2019-11-03 22:11:45 +01:00
from async_worker.async_worker import AsyncTask
from pyrogram.api.types import ChannelMessagesFilterEmpty
from pyrogram.api.types.updates import ChannelDifference, ChannelDifferenceTooLong, ChannelDifferenceEmpty
from pyrogram.api.functions.updates.get_channel_difference import GetChannelDifference
from pyrogram.client.types.messages_and_media import Message as MessagePyrogram
2019-11-04 15:08:33 +01:00
class JsonSerializerFixes:
@staticmethod
def user(obj):
obj.type = "private"
return obj
@staticmethod
def user_type(obj):
obj.type = "channel" if obj.id < 0 else "private"
return obj
class JsonSerializer:
fixes = {
"from_user": {
"new_name": "from",
"patch": JsonSerializerFixes.user
},
"user": {
"new_name": "user",
"patch": JsonSerializerFixes.user_type
}
}
@staticmethod
def default(obj):
if isinstance(obj, bytes):
return repr(obj)
cls = JsonSerializer
result = {}
for name in filter(lambda x: not x.startswith("_"), obj.__dict__):
value = getattr(obj, name)
if value is None:
continue
if name in cls.fixes:
value = cls.fixes[name]["patch"](value)
name = cls.fixes[name]["new_name"]
result[name] = value
return result
2019-11-03 22:11:45 +01:00
class ChannelHistoryReadTask(AsyncTask):
channel: pyrogram.Chat
client: pyrogram.Client
pts: int
webhook: str
2019-11-04 15:08:33 +01:00
http: aiohttp.ClientSession
2019-11-03 22:11:45 +01:00
def setup(self, client: pyrogram.Client, channel: pyrogram.Chat, webhook: str):
self.client = client
self.channel = channel
self.pts = False
self.webhook = webhook
2019-11-04 15:08:33 +01:00
self.http = aiohttp.ClientSession()
2019-11-03 22:11:45 +01:00
async def process(self):
response = await self.client.send(
GetChannelDifference(
channel=self.channel,
filter=ChannelMessagesFilterEmpty(),
pts=self.pts if self.pts else 0xFFFFFFF,
limit=0xFFFFFFF,
force=True
)
)
if isinstance(response, ChannelDifference):
self.pts = response.pts
users = {i.id: i for i in response.users}
chats = {i.id: i for i in response.chats}
for message in response.new_messages:
message = await MessagePyrogram._parse(self.client, message, users, chats)
2019-11-04 15:08:33 +01:00
message = {"update_id": 1, "message": message}
data = json.dumps(
message,
default=JsonSerializer.default,
ensure_ascii=True,
allow_nan=False,
check_circular=True,
sort_keys=False
)
result = await self.http.post(
self.webhook,
data=data,
headers=[
("Content-Type", "application/json")
]
)
2019-11-03 22:11:45 +01:00
2019-11-04 15:38:37 +01:00
await result.read()
result.close()
2019-11-03 22:11:45 +01:00
if not response.final:
return 1
return response.timeout
if isinstance(response, ChannelDifferenceEmpty):
self.pts = response.pts
return response.timeout
if isinstance(response, ChannelDifferenceTooLong):
return False