2
0
mirror of https://github.com/Nick80835/microbot synced 2025-08-23 02:28:31 +00:00
microbot/ubot/command_handler.py

312 lines
12 KiB
Python
Raw Normal View History

2020-05-16 17:52:29 -04:00
import asyncio
2020-06-29 15:29:20 -04:00
from random import randint
from re import DOTALL, IGNORECASE, escape, search
2020-10-04 13:43:57 -04:00
from traceback import format_exc, print_exc
2023-10-16 09:46:31 -04:00
2020-10-03 11:27:11 -04:00
from telethon import events
2023-10-16 09:46:31 -04:00
from .database import ChatWrapper
2020-06-15 13:04:35 -04:00
from .fixes import inline_photos
2023-10-16 09:46:31 -04:00
class CommandHandler():
2021-02-16 19:08:52 -05:00
pattern_template = "^{0}({1})(?: |\n|$|_|@{2}(?: |\n|$|_))(.*)"
simple_pattern_template = "^({0})(?: |\n|$)(.*)"
raw_pattern_template = "{0}"
2020-07-18 15:11:09 -04:00
incoming_commands = []
inline_photo_commands = []
inline_article_commands = []
callback_queries = []
2021-08-26 10:47:36 -04:00
def __init__(self, client, settings, loader, logger):
self.username = client.loop.run_until_complete(client.get_me()).username
self.settings = settings
2020-06-23 15:28:09 -04:00
self.loader = loader
self.db = loader.db
2021-08-26 10:47:36 -04:00
self.logger = logger
2021-03-23 20:01:27 -04:00
self.hard_prefix = self.settings.get_list("hard_cmd_prefix") or ["/"]
2021-01-27 17:03:04 -05:00
client.add_event_handler(self.report_incoming_excepts, events.NewMessage(incoming=True, forwards=False, func=lambda e: e.raw_text))
2020-05-25 14:04:56 -04:00
client.add_event_handler(self.handle_inline, events.InlineQuery())
client.add_event_handler(self.handle_callback_query, events.CallbackQuery())
2023-10-16 09:46:31 -04:00
async def report_incoming_excepts(self, event):
try:
await self.handle_incoming(event)
except:
await event.client.send_message(int(self.settings.get_list("owner_id")[0]), str(format_exc()))
async def handle_incoming(self, event):
chat_db = ChatWrapper(self.db.get_chat(event.chat.id))
chat_prefix = chat_db.prefix
2023-10-16 09:46:31 -04:00
for command in self.incoming_commands:
if command.simple_pattern:
pattern_match = search(self.simple_pattern_template.format(command.pattern + command.pattern_extra), event.raw_text, IGNORECASE|DOTALL)
elif command.raw_pattern:
pattern_match = search(self.raw_pattern_template.format(command.pattern + command.pattern_extra), event.raw_text, IGNORECASE|DOTALL)
2020-06-26 19:17:50 -04:00
else:
if command.not_disableable:
prefix_list = self.hard_prefix + [chat_prefix] + ["/"]
else:
prefix_list = self.hard_prefix + [chat_prefix]
pattern_match = search(self.pattern_template.format(f"({'|'.join([escape(i) for i in prefix_list])})", command.pattern + command.pattern_extra, self.username), event.raw_text, IGNORECASE|DOTALL)
2020-04-17 16:47:19 -04:00
if pattern_match:
if not await self.check_privs(event, command, chat_db):
2020-07-18 15:11:09 -04:00
return
if command.pass_nsfw:
event.nsfw_disabled = not chat_db.nsfw_enabled
2020-10-03 15:34:26 -04:00
if command.raw_pattern:
event.command = command.pattern
else:
event.command = pattern_match.groups()[1]
2020-08-24 12:07:23 -04:00
if event.chat and not command.not_disableable and event.command in chat_db.disabled_commands():
2020-08-24 12:07:23 -04:00
return
2020-10-03 15:34:26 -04:00
if not command.raw_pattern:
event.args = pattern_match.groups()[-1].strip()
if command.simple_pattern:
event.other_args = pattern_match.groups()[1:-1]
else:
event.other_args = pattern_match.groups()[2:-1]
2021-03-23 20:07:59 -04:00
event.prefix = pattern_match.groups()[0]
2020-10-03 15:34:26 -04:00
2020-04-17 16:47:19 -04:00
event.pattern_match = pattern_match
event.extra = command.extra
event.object = command
event.chat_db = chat_db
2023-10-16 09:46:31 -04:00
2021-08-26 10:47:36 -04:00
self.logger.info(f"Handling command '{event.command}' for {event.sender_id}")
await self.execute_command(event, command)
2020-05-16 17:52:29 -04:00
2020-05-25 14:04:56 -04:00
async def handle_inline(self, event):
for command in self.inline_photo_commands:
pattern_match = search(self.simple_pattern_template.format(command.pattern + command.pattern_extra), event.text, IGNORECASE|DOTALL)
2020-05-16 17:52:29 -04:00
2020-05-25 14:04:56 -04:00
if pattern_match:
2020-10-01 20:55:46 -04:00
if self.is_blacklisted(event, True) and not self.is_owner(event) and not self.is_sudo(event):
return
await self.handle_inline_photo(event, pattern_match, command)
2020-05-25 14:04:56 -04:00
return
2020-05-21 13:53:11 -04:00
for command in self.inline_article_commands:
pattern_match = search(self.simple_pattern_template.format(command.pattern + command.pattern_extra), event.text, IGNORECASE|DOTALL)
2020-05-21 13:53:11 -04:00
if pattern_match:
2020-10-01 20:55:46 -04:00
if self.is_blacklisted(event, True) and not self.is_owner(event) and not self.is_sudo(event):
return
await self.handle_inline_article(event, pattern_match, command)
2020-05-25 14:04:56 -04:00
return
2020-05-16 17:52:29 -04:00
2020-05-25 14:04:56 -04:00
await self.fallback_inline(event)
2020-05-21 13:53:11 -04:00
async def handle_inline_photo(self, event, pattern_match, command):
2020-05-25 14:04:56 -04:00
builder = event.builder
event.pattern_match = pattern_match
event.args = pattern_match.groups()[-1]
2020-07-16 13:41:42 -04:00
event.other_args = pattern_match.groups()[1:-1]
event.command = pattern_match.groups()[0]
2020-10-03 17:31:15 -04:00
event.extra = command.extra
event.object = command
2020-12-08 21:04:48 -05:00
event.parse_mode = command.parse_mode
2020-05-21 13:53:11 -04:00
photo_list = await command.function(event)
2020-05-21 13:53:11 -04:00
2020-06-15 13:04:35 -04:00
if not photo_list:
2020-05-25 14:04:56 -04:00
return
2020-05-21 13:53:11 -04:00
2020-05-25 14:04:56 -04:00
photo_coros = []
2020-05-16 17:52:29 -04:00
2020-06-15 13:04:35 -04:00
for photo in photo_list:
2020-05-25 14:04:56 -04:00
try:
2020-06-15 13:04:35 -04:00
if isinstance(photo, list):
2020-12-08 21:04:48 -05:00
photo_coros += [
self.try_coro(inline_photos.photo(
event.client,
photo[0],
text=photo[1],
parse_mode=event.parse_mode
))
]
2020-06-15 13:04:35 -04:00
else:
photo_coros += [self.try_coro(builder.photo(photo))]
2020-05-25 14:04:56 -04:00
except:
2020-10-04 13:43:57 -04:00
print_exc()
2020-05-25 14:04:56 -04:00
if photo_coros:
photos = await asyncio.gather(*photo_coros)
else:
return
2020-05-16 17:52:29 -04:00
2020-05-25 14:04:56 -04:00
try:
2020-05-25 14:38:53 -04:00
await event.answer([i for i in photos if i], gallery=True)
2020-05-25 14:04:56 -04:00
except:
2020-10-04 13:43:57 -04:00
print_exc()
2020-05-25 14:04:56 -04:00
async def handle_inline_article(self, event, pattern_match, command):
2020-05-25 14:04:56 -04:00
builder = event.builder
event.pattern_match = pattern_match
event.args = pattern_match.groups()[-1]
event.other_args = pattern_match.groups()[1:-1]
event.command = pattern_match.groups()[0]
2020-10-03 17:31:15 -04:00
event.extra = command.extra
event.object = command
event.link_preview = command.link_preview
2020-12-08 21:04:48 -05:00
event.parse_mode = command.parse_mode
2020-05-25 14:04:56 -04:00
result_list = await command.function(event)
2020-05-25 14:04:56 -04:00
if not result_list:
return
articles = []
for result in result_list:
2020-05-24 12:10:54 -04:00
try:
2020-12-08 21:04:48 -05:00
articles += [
await builder.article(
title=result["title"],
description=result["description"],
text=result["text"],
link_preview=event.link_preview,
parse_mode=event.parse_mode
)
]
2020-05-24 12:10:54 -04:00
except:
2020-10-04 13:43:57 -04:00
print_exc()
2020-05-16 17:52:29 -04:00
2020-05-25 14:04:56 -04:00
try:
await event.answer([i for i in articles if i])
except:
2020-10-04 13:43:57 -04:00
print_exc()
2020-05-25 14:04:56 -04:00
async def handle_callback_query(self, event):
data_str = event.data.decode("utf-8")
data_id = data_str.split("*")[0]
data_data = data_str.lstrip(data_id + "*")
for command in self.callback_queries:
if command.data_id == data_id:
event.command = data_id
event.args = data_data
event.extra = command.extra
event.object = command
try:
await command.function(event)
except Exception as exception:
await event.reply(f"An error occurred in **{command.function.__name__}**: `{exception}`")
2020-10-04 13:43:57 -04:00
print_exc()
2020-05-25 14:04:56 -04:00
async def fallback_inline(self, event):
2020-06-16 17:00:09 -04:00
defaults_list = self.inline_photo_commands + self.inline_article_commands
2020-05-25 14:04:56 -04:00
try:
await event.answer([await event.builder.article(title=command.pattern, text=f"{self.loader.prefix()}{command.default}") for command in defaults_list if command.default])
2020-05-25 14:04:56 -04:00
except:
2020-10-04 13:43:57 -04:00
print_exc()
2020-05-25 14:04:56 -04:00
2020-05-21 13:53:11 -04:00
async def try_coro(self, coro):
2020-05-16 17:52:29 -04:00
try:
2020-05-21 13:53:11 -04:00
return await coro
2020-05-16 17:52:29 -04:00
except:
2020-05-21 13:53:11 -04:00
return
async def execute_command(self, event, command):
2020-07-18 15:11:09 -04:00
try:
if command.locking:
if command.lock_reason:
await event.reply(f"That command is currently locked: {command.lock_reason}")
2020-07-18 15:11:09 -04:00
return
if command.chance and randint(0, 100) <= command.chance or not command.chance:
2020-10-03 11:27:11 -04:00
command.lock_reason = f"In use by **{event.sender_id}** (`{event.raw_text}`)"
await command.function(event)
command.lock_reason = None
elif command.user_locking:
2020-10-03 11:27:11 -04:00
if event.sender_id in command.locked_users:
2021-02-11 10:47:08 -05:00
await event.reply("Please don't spam that command.")
2020-07-18 15:11:09 -04:00
return
if command.chance and randint(0, 100) <= command.chance or not command.chance:
2020-10-03 11:27:11 -04:00
command.locked_users.append(event.sender_id)
await command.function(event)
2020-10-03 11:27:11 -04:00
command.locked_users.remove(event.sender_id)
2020-07-18 15:11:09 -04:00
else:
if command.chance and randint(0, 100) <= command.chance or not command.chance:
await command.function(event)
2020-07-18 15:11:09 -04:00
except Exception as exception:
command.lock_reason = None
2020-07-18 15:11:09 -04:00
2020-10-03 11:27:11 -04:00
if event.sender_id in command.locked_users:
command.locked_users.remove(event.sender_id)
2020-07-18 15:11:09 -04:00
2021-06-22 08:49:37 -04:00
if not command.silent_bail:
try:
await event.reply(f"An error occurred in **{command.function.__name__}**: `{exception}`")
except:
pass
2020-10-04 13:43:57 -04:00
print_exc()
2020-07-18 15:11:09 -04:00
async def check_privs(self, event, command, chat_db):
if self.is_blacklisted(event) and not self.is_owner(event) and not self.is_sudo(event):
return False
2021-03-23 20:01:27 -04:00
if command.no_private and event.is_private:
await event.reply("That command can't be used in private!")
return False
if command.owner and not self.is_owner(event):
if not command.silent_bail:
await event.reply("You lack the permissions to use that command!")
return False
if command.sudo and not self.is_sudo(event) and not self.is_owner(event):
if not command.silent_bail:
await event.reply("You lack the permissions to use that command!")
return False
2020-11-27 10:20:05 -05:00
if command.admin:
2020-11-30 23:38:03 -05:00
if event.chat and event.sender_id:
2020-11-27 10:20:05 -05:00
if event.is_private or not (await event.client.get_permissions(event.chat, event.sender_id)).is_admin and not self.is_sudo(event) and not self.is_owner(event):
if not command.silent_bail:
await event.reply("You lack the permissions to use that command!")
2020-11-27 10:20:05 -05:00
return False
if event.chat and command.nsfw and not chat_db.nsfw_enabled:
if not command.silent_bail:
await event.reply(command.nsfw_warning or "NSFW commands are disabled in this chat!")
2020-06-20 10:13:45 -04:00
return False
2020-06-12 13:49:57 -04:00
if event.chat and command.fun and not chat_db.fun_enabled:
2020-08-04 07:06:11 -04:00
return False
2020-07-18 15:11:09 -04:00
return True
def is_owner(self, event):
return str(event.sender_id) in self.settings.get_list("owner_id")
2020-07-18 15:11:09 -04:00
def is_sudo(self, event):
return event.sender_id in self.db.get_sudo_list()
2020-07-18 15:11:09 -04:00
2020-06-13 20:30:43 -04:00
def is_blacklisted(self, event, inline=False):
if inline:
user_id = event.query.user_id
else:
2020-10-01 20:55:46 -04:00
user_id = event.sender_id
2020-06-13 20:30:43 -04:00
return user_id in self.db.get_blacklist_list()