mirror of
https://github.com/RYGhub/royalnet.git
synced 2024-11-27 13:34:28 +00:00
Bots can now getUpdates
This commit is contained in:
parent
ea15416356
commit
789d94a3f9
1 changed files with 260 additions and 4 deletions
264
telegram.py
264
telegram.py
|
@ -2,26 +2,63 @@ import asyncio
|
||||||
loop = asyncio.get_event_loop()
|
loop = asyncio.get_event_loop()
|
||||||
import aiohttp
|
import aiohttp
|
||||||
import async_timeout
|
import async_timeout
|
||||||
|
import datetime
|
||||||
|
|
||||||
class TelegramError(Exception):
|
class TelegramAPIError(Exception):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
class UpdateError(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
class Bot:
|
class Bot:
|
||||||
def __init__(self, token):
|
def __init__(self, token):
|
||||||
self.token = token
|
self.token = token
|
||||||
self.user_data = None
|
self.user_data = None
|
||||||
self.updates = list()
|
self.updates = list()
|
||||||
self.users = list()
|
|
||||||
self.chats = list()
|
self.chats = list()
|
||||||
|
self.commands = list()
|
||||||
|
self.offset = 0
|
||||||
# Update user_data
|
# Update user_data
|
||||||
loop.run_until_complete(self.update_bot_data())
|
loop.run_until_complete(self.update_bot_data())
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return f"<Bot {self.user_data.first_name}>"
|
||||||
|
|
||||||
|
def __hash__(self):
|
||||||
|
return hash(self.token)
|
||||||
|
|
||||||
async def update_bot_data(self):
|
async def update_bot_data(self):
|
||||||
"""Update self.user_data with the latest information from /getMe."""
|
"""Update self.user_data with the latest information from /getMe."""
|
||||||
data = await self.api_request("getMe")
|
data = await self.api_request("getMe")
|
||||||
self.user_data = User(data)
|
self.user_data = User(data)
|
||||||
|
|
||||||
|
async def get_updates(self):
|
||||||
|
"""Get the latest updates from the Telegram API with /getUpdates."""
|
||||||
|
data = await self.api_request("getUpdates", offset=self.offset, timeout=0)
|
||||||
|
for update in data:
|
||||||
|
try:
|
||||||
|
self.updates.append(Update(update))
|
||||||
|
except NotImplementedError:
|
||||||
|
pass
|
||||||
|
if len(data) > 0:
|
||||||
|
self.offset = self.updates[-1].update_id + 1
|
||||||
|
|
||||||
|
def parse_update(self):
|
||||||
|
"""Parse the first update in the list."""
|
||||||
|
update = self.updates[0]
|
||||||
|
if update.message.chat not in self.chats:
|
||||||
|
self.chats.append(update.message.chat)
|
||||||
|
chat = self.find_chat(update.message.chat.chat_id)
|
||||||
|
if update.message.sent_from not in chat.users:
|
||||||
|
chat.users.append(update.message.sent_from)
|
||||||
|
chat.messages.append(update.message)
|
||||||
|
del self.updates[0]
|
||||||
|
|
||||||
|
def find_chat(self, chat_id):
|
||||||
|
for chat in self.chats:
|
||||||
|
if chat.chat_id == chat_id:
|
||||||
|
return chat
|
||||||
|
|
||||||
async def api_request(self, endpoint, **params):
|
async def api_request(self, endpoint, **params):
|
||||||
"""Send a request to the Telegram API at the specified endpoint."""
|
"""Send a request to the Telegram API at the specified endpoint."""
|
||||||
# Request timeout is 10 seconds.
|
# Request timeout is 10 seconds.
|
||||||
|
@ -33,16 +70,81 @@ class Bot:
|
||||||
async with session.request("GET", f"https://api.telegram.org/bot{token}/{endpoint}", params=params) as response:
|
async with session.request("GET", f"https://api.telegram.org/bot{token}/{endpoint}", params=params) as response:
|
||||||
# Check for errors in the request
|
# Check for errors in the request
|
||||||
if response.status != 200:
|
if response.status != 200:
|
||||||
raise TelegramError(f"Request returned {response.status} {response.reason}")
|
raise TelegramAPIError(f"Request returned {response.status} {response.reason}")
|
||||||
# Parse the json data as soon it's ready
|
# Parse the json data as soon it's ready
|
||||||
data = await response.json()
|
data = await response.json()
|
||||||
# Check for errors in the response
|
# Check for errors in the response
|
||||||
if not data["ok"]:
|
if not data["ok"]:
|
||||||
error = data["description"]
|
error = data["description"]
|
||||||
raise TelegramError(f"Response returned an error: {error}")
|
raise TelegramAPIError(f"Response returned an error: {error}")
|
||||||
# Return a dictionary containing the data
|
# Return a dictionary containing the data
|
||||||
return data["result"]
|
return data["result"]
|
||||||
|
|
||||||
|
|
||||||
|
class Update:
|
||||||
|
def __init__(self, upd_dict):
|
||||||
|
self.update_id = upd_dict["update_id"]
|
||||||
|
if "message" in upd_dict:
|
||||||
|
self.message = Message(upd_dict["message"])
|
||||||
|
elif "edited_message" in upd_dict:
|
||||||
|
self.message = Message(upd_dict["edited_message"])
|
||||||
|
elif "channel_post" in upd_dict:
|
||||||
|
self.message = Message(upd_dict["channel_post"])
|
||||||
|
elif "edited_channel_post" in upd_dict:
|
||||||
|
self.message = Message(upd_dict["edited_channel_post"])
|
||||||
|
else:
|
||||||
|
raise NotImplementedError("No inline support yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Chat:
|
||||||
|
def __init__(self, chat_dict):
|
||||||
|
self.chat_id = chat_dict["id"]
|
||||||
|
self.type = chat_dict["type"]
|
||||||
|
self.users = list()
|
||||||
|
self.admins = list()
|
||||||
|
self.messages = list()
|
||||||
|
self.chat_photo = None
|
||||||
|
if self.type == "private":
|
||||||
|
self.first_name = chat_dict["first_name"]
|
||||||
|
if "last_name" in chat_dict:
|
||||||
|
self.last_name = chat_dict["last_name"]
|
||||||
|
else:
|
||||||
|
self.last_name = None
|
||||||
|
if "username" in chat_dict:
|
||||||
|
self.username = chat_dict["username"]
|
||||||
|
else:
|
||||||
|
self.username = None
|
||||||
|
self.title = f"{self.first_name} {self.last_name}"
|
||||||
|
self.everyone_is_admin = True
|
||||||
|
elif self.type == "group" or self.type == "supergroup" or self.type == "channel":
|
||||||
|
self.first_name = None
|
||||||
|
self.last_name = None
|
||||||
|
if self.type == "supergroup" or self.type == "channel":
|
||||||
|
self.everyone_is_admin = False
|
||||||
|
if "username" in chat_dict:
|
||||||
|
self.username = chat_dict["username"]
|
||||||
|
else:
|
||||||
|
self.username = None
|
||||||
|
else:
|
||||||
|
self.everyone_is_admin = chat_dict["all_members_are_administrators"]
|
||||||
|
self.username = None
|
||||||
|
self.title = chat_dict["title"]
|
||||||
|
else:
|
||||||
|
raise UpdateError(f"Unknown message type: {self.type}")
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return f"<{self.type} Chat {self.title}>"
|
||||||
|
|
||||||
|
def __hash__(self):
|
||||||
|
return self.chat_id
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
if isinstance(other, Chat):
|
||||||
|
return self.chat_id == other.chat_id
|
||||||
|
else:
|
||||||
|
TypeError("Can't compare Chat to a different object.")
|
||||||
|
|
||||||
|
|
||||||
class User:
|
class User:
|
||||||
def __init__(self, user_dict):
|
def __init__(self, user_dict):
|
||||||
self.user_id = user_dict["id"]
|
self.user_id = user_dict["id"]
|
||||||
|
@ -55,3 +157,157 @@ class User:
|
||||||
self.username = user_dict["username"]
|
self.username = user_dict["username"]
|
||||||
else:
|
else:
|
||||||
self.username = None
|
self.username = None
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
if self.username is not None:
|
||||||
|
return f"<User {self.username}>"
|
||||||
|
else:
|
||||||
|
return f"<User {self.user_id}>"
|
||||||
|
|
||||||
|
def __hash__(self):
|
||||||
|
return self.user_id
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
if isinstance(other, User):
|
||||||
|
return self.user_id == other.user_id
|
||||||
|
else:
|
||||||
|
TypeError("Can't compare User to a different object.")
|
||||||
|
|
||||||
|
|
||||||
|
class Message:
|
||||||
|
def __init__(self, msg_dict):
|
||||||
|
self.msg_id = msg_dict["message_id"]
|
||||||
|
self.date = datetime.datetime.fromtimestamp(msg_dict["date"])
|
||||||
|
self.chat = Chat(msg_dict["chat"])
|
||||||
|
if "from" in msg_dict:
|
||||||
|
self.sent_from = User(msg_dict["from"])
|
||||||
|
else:
|
||||||
|
self.sent_from = None
|
||||||
|
self.forwarded = "forward_date" in msg_dict
|
||||||
|
if self.forwarded:
|
||||||
|
if "forward_from" in msg_dict:
|
||||||
|
self.original_sender = User(msg_dict["forward_from"])
|
||||||
|
elif "forward_from_chat" in msg_dict:
|
||||||
|
self.original_sender = Chat(msg_dict["forward_from_chat"])
|
||||||
|
# TODO: Add forward_from_message_id
|
||||||
|
if "reply_to_message" in msg_dict:
|
||||||
|
self.is_reply_to = Message(msg_dict["reply_to_message"])
|
||||||
|
else:
|
||||||
|
self.is_reply_to = None
|
||||||
|
if "text" in msg_dict:
|
||||||
|
self.content = msg_dict["text"]
|
||||||
|
# TODO: Check for MessageEntities
|
||||||
|
elif "audio" in msg_dict:
|
||||||
|
self.content = Audio(msg_dict["audio"])
|
||||||
|
elif "document" in msg_dict:
|
||||||
|
self.content = Document(msg_dict["document"])
|
||||||
|
elif "game" in msg_dict:
|
||||||
|
self.content = Game(msg_dict["game"])
|
||||||
|
elif "photo" in msg_dict:
|
||||||
|
self.content = Photo(msg_dict["photo"])
|
||||||
|
elif "sticker" in msg_dict:
|
||||||
|
self.content = Sticker(msg_dict["sticker"])
|
||||||
|
elif "video" in msg_dict:
|
||||||
|
self.content = Video(msg_dict["video"])
|
||||||
|
elif "voice" in msg_dict:
|
||||||
|
self.content = Voice(msg_dict["voice"])
|
||||||
|
elif "contact" in msg_dict:
|
||||||
|
self.content = Contact(msg_dict["contact"])
|
||||||
|
elif "location" in msg_dict:
|
||||||
|
self.content = Location(msg_dict["location"])
|
||||||
|
elif "venue" in msg_dict:
|
||||||
|
self.content = Venue(msg_dict["venue"])
|
||||||
|
elif "new_chat_member" in msg_dict:
|
||||||
|
self.content = ServiceMessage("new_chat_member", User(msg_dict["new_chat_member"]))
|
||||||
|
elif "left_chat_member" in msg_dict:
|
||||||
|
self.content = ServiceMessage("left_chat_member", User(msg_dict["left_chat_member"]))
|
||||||
|
elif "new_chat_title" in msg_dict:
|
||||||
|
self.content = ServiceMessage("new_chat_title", msg_dict["new_chat_title"])
|
||||||
|
elif "new_chat_photo" in msg_dict:
|
||||||
|
self.content = ServiceMessage("new_chat_photo", Photo(msg_dict["new_chat_photo"]))
|
||||||
|
elif "delete_chat_photo" in msg_dict:
|
||||||
|
self.content = ServiceMessage("delete_chat_photo")
|
||||||
|
elif "group_chat_created" in msg_dict:
|
||||||
|
self.content = ServiceMessage("group_chat_created")
|
||||||
|
elif "supergroup_chat_created" in msg_dict:
|
||||||
|
self.content = ServiceMessage("supergroup_chat_created")
|
||||||
|
elif "channel_chat_created" in msg_dict:
|
||||||
|
self.content = ServiceMessage("channel_chat_created")
|
||||||
|
elif "migrate_to_chat_id" in msg_dict:
|
||||||
|
self.content = ServiceMessage("migrate_to_chat_id", msg_dict["migrate_to_chat_id"])
|
||||||
|
elif "migrate_from_chat_id" in msg_dict:
|
||||||
|
self.content = ServiceMessage("migrate_from_chat_id", msg_dict["migrate_from_chat_id"])
|
||||||
|
elif "pinned_message" in msg_dict:
|
||||||
|
self.content = ServiceMessage("pinned_message", Message(msg_dict["pinned_message"]))
|
||||||
|
else:
|
||||||
|
raise UpdateError("Message doesn't contain anything.")
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
if isinstance(self.content, str):
|
||||||
|
return f"<Message: {self.content}>"
|
||||||
|
else:
|
||||||
|
return f"<Message containing {type(self.content)}>"
|
||||||
|
|
||||||
|
|
||||||
|
class ServiceMessage:
|
||||||
|
def __init__(self, msg_type, extra=None):
|
||||||
|
self.type = msg_type
|
||||||
|
self.content = extra
|
||||||
|
|
||||||
|
|
||||||
|
class Audio:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Document:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Game:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Photo:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Sticker:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Video:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Voice:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Contact:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Location:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
class Venue:
|
||||||
|
def __init__(self, init_dict):
|
||||||
|
raise NotImplementedError("Not yet.")
|
||||||
|
|
||||||
|
|
||||||
|
b = Bot("369842636:AAGfCd1ZNIgmU5w2ltmpSjMFt2c7O8tHvaU")
|
||||||
|
while True:
|
||||||
|
loop.run_until_complete(b.get_updates())
|
||||||
|
while len(b.updates) > 0:
|
||||||
|
b.parse_update()
|
||||||
|
print("Completed.")
|
Loading…
Reference in a new issue