def del_blacklist(update: Update, context: CallbackContext): chat = update.effective_chat message = update.effective_message to_match = extract_text(message) msg = update.effective_message if not to_match: return chat_filters = sql.get_chat_blacklist(chat.id) for trigger in chat_filters: pattern = r"( |^|[^\w])" + trigger + r"( |$|[^\w])" match = regex_searcher(pattern, to_match) if not match: # Skip to next item in blacklist continue if match: try: message.delete() except BadRequest as excp: if excp.message == "Mensaje para eliminar no encontrado": pass else: LOGGER.exception( "Error al eliminar el mensaje de la lista negra.") break
def shell(update: Update, context: CallbackContext): message = update.effective_message cmd = message.text.split(" ", 1) if len(cmd) == 1: message.reply_text("No se dio ningún comando para ejecutar.") return cmd = cmd[1] process = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.PIPE, shell=True) stdout, stderr = process.communicate() reply = "" stderr = stderr.decode() stdout = stdout.decode() if stdout: reply += f"*Stdout*\n`{stdout}`\n" LOGGER.info(f"Shell - {cmd} - {stdout}") if stderr: reply += f"*Stderr*\n`{stderr}`\n" LOGGER.error(f"Shell - {cmd} - {stderr}") if len(reply) > 3000: with open("shell_output.txt", "w") as file: file.write(reply) with open("shell_output.txt", "rb") as doc: context.bot.send_document( document=doc, filename=doc.name, reply_to_message_id=message.message_id, chat_id=message.chat_id, ) else: message.reply_text(reply, parse_mode=ParseMode.MARKDOWN)
def glog_action(update: Update, context: CallbackContext, *args, **kwargs): result = func(update, context, *args, **kwargs) chat = update.effective_chat message = update.effective_message if result: datetime_fmt = "%H:%M - %d-%m-%Y" result += "\n<b>Evento Registrado</b>: <code>{}</code>".format( datetime.utcnow().strftime(datetime_fmt)) if message.chat.type == chat.SUPERGROUP and message.chat.username: result += f'\n<b>Link:</b> <a href="https://t.me/{chat.username}/{message.message_id}">click acá</a>' log_chat = str(GLOBAL_LOGS) if log_chat: send_log(context, log_chat, chat.id, result) elif result == "" or not result: pass else: LOGGER.warning( "%s was set as loggable to gbanlogs, but had no return statement.", func, ) return result
def get_user_id(username): # ensure valid userid if len(username) <= 5: return None if username.startswith("@"): username = username[1:] users = sql.get_userid_by_name(username) if not users: return None elif len(users) == 1: return users[0].user_id else: for user_obj in users: try: userdat = dispatcher.bot.get_chat(user_obj.user_id) if userdat.username == username: return userdat.id except BadRequest as excp: if excp.message == "Chat not found": pass else: LOGGER.exception("Error al extraer el ID de usuario") return None
def log_action( update: Update, context: CallbackContext, job_queue: JobQueue = None, *args, **kwargs, ): if not job_queue: result = func(update, context, *args, **kwargs) else: result = func(update, context, job_queue, *args, **kwargs) chat = update.effective_chat message = update.effective_message if result: datetime_fmt = "%H:%M - %d-%m-%Y" result += f"\n<b>Evento Registrado</b>: <code>{datetime.utcnow().strftime(datetime_fmt)}</code>" if message.chat.type == chat.SUPERGROUP and message.chat.username: result += f'\n<b>Link:</b> <a href="https://t.me/{chat.username}/{message.message_id}">Click acá</a>' log_chat = sql.get_chat_log_channel(chat.id) if log_chat: send_log(context, log_chat, chat.id, result) elif result == "" or not result: pass else: LOGGER.warning( "%s was set as loggable, but had no return statement.", func) return result
def __list_all_modules(): import glob from os.path import basename, dirname, isfile # This generates a list of modules in this folder for the * in __main__ to work. mod_paths = glob.glob(dirname(__file__) + "/*.py") all_modules = [ basename(f)[:-3] for f in mod_paths if isfile(f) and f.endswith(".py") and not f.endswith("__init__.py") ] if LOAD or NO_LOAD: to_load = LOAD if to_load: if not all( any(mod == module_name for module_name in all_modules) for mod in to_load): LOGGER.error("Invalid loadorder names. Quitting.") quit(1) all_modules = sorted(set(all_modules) - set(to_load)) to_load = list(all_modules) + to_load else: to_load = all_modules if NO_LOAD: LOGGER.info("Not loading: {}".format(NO_LOAD)) return [item for item in to_load if item not in NO_LOAD] return to_load return all_modules
def get_exception(excp, filt, chat): if excp.message == "Protocolo de URL no admitido": return "Parece que intentas utilizar un protocolo de URL no compatible. Telegram no admite botones para algunos protocolos, como tg://. Por favor, inténtalo de nuevo" elif excp.message == "Reply message not found": return "noreply" else: LOGGER.warning("Message %s could not be parsed", str(filt.reply)) LOGGER.exception("Could not parse filter %s in chat %s", str(filt.keyword), str(chat.id)) return "Estos datos no se pudieron enviar porque están formateados incorrectamente."
async def whois(client, message): buscando = await message.reply_text(text="Buscando...") cmd = message.command if not message.reply_to_message and len(cmd) == 1: get_user = message.from_user.id elif len(cmd) == 1: get_user = message.reply_to_message.from_user.id elif len(cmd) > 1: get_user = cmd[1] try: get_user = int(cmd[1]) except ValueError: pass try: user = await client.get_users(get_user) except PeerIdInvalid: await buscando.edit("No conozco a este usuario.") return desc = await client.get_chat(get_user) desc = desc.description user_pic = await client.get_profile_photos(user.id) pic_count = await client.get_profile_photos_count(user.id) if not user.photo: await message.reply( WHOIS.format( user_id=user.id, full_name=FullName(user), userid=user.id, username=user.username if user.username else "`Sin alias`", last_online=LastOnline(user), ), disable_web_page_preview=True, ) else: await client.download_media(user_pic[0], file_name=f"./{user.id}.png") await message.reply_document( document=open(f"{user.id}.png", "rb"), caption=WHOIS_PIC.format( user_id=user.id, full_name=FullName(user), userid=user.id, username=user.username if user.username else "`Sin alias`", last_online=LastOnline(user), profile_pics=pic_count, reply_to_message_id=ReplyCheck(message), ), ) try: os.remove(f"./{user.id}.png") except: LOGGER.exception("Error al remover foto de perfil") try: await buscando.delete() except: LOGGER.exception("Error al eliminar mensaje Buscando")
def send(msg, bot, update): if len(str(msg)) > 2000: with io.BytesIO(str.encode(msg)) as out_file: out_file.name = "output.txt" bot.send_document(chat_id=update.effective_chat.id, document=out_file) else: LOGGER.info(f"Salida: '{msg}'") bot.send_message( chat_id=update.effective_chat.id, text=f"`{msg}`", parse_mode=ParseMode.MARKDOWN, )
def main(): test_handler = CommandHandler("test", test, run_async=True) start_handler = CommandHandler("start", start, run_async=True) help_handler = CommandHandler("help", get_help, run_async=True) help_callback_handler = CallbackQueryHandler(help_button, pattern=r"help_.*", run_async=True) settings_handler = CommandHandler("settings", get_settings, run_async=True) settings_callback_handler = CallbackQueryHandler(settings_button, pattern=r"stngs_", run_async=True) donate_handler = CommandHandler("donate", donate) migrate_handler = MessageHandler(Filters.status_update.migrate, migrate_chats) # dispatcher.add_handler(test_handler) dispatcher.add_handler(start_handler) dispatcher.add_handler(help_handler) dispatcher.add_handler(settings_handler) dispatcher.add_handler(help_callback_handler) dispatcher.add_handler(settings_callback_handler) dispatcher.add_handler(migrate_handler) dispatcher.add_handler(donate_handler) dispatcher.add_error_handler(error_callback) if WEBHOOK: LOGGER.info("Usando webhooks.") updater.start_webhook(listen="0.0.0.0", port=PORT, url_path=TOKEN) if CERT_PATH: updater.bot.set_webhook(url=URL + TOKEN, certificate=open(CERT_PATH, "rb")) else: updater.bot.set_webhook(url=URL + TOKEN) else: LOGGER.info("Usando long polling.") updater.start_polling(timeout=15, read_latency=4, drop_pending_updates=True) if len(argv) not in (1, 3, 4): telethn.disconnect() else: telethn.run_until_disconnected() updater.idle()
def migrate_chats(update: Update, context: CallbackContext): msg = update.effective_message # type: Optional[Message] if msg.migrate_to_chat_id: old_chat = update.effective_chat.id new_chat = msg.migrate_to_chat_id elif msg.migrate_from_chat_id: old_chat = msg.migrate_from_chat_id new_chat = update.effective_chat.id else: return LOGGER.info("Migrating from %s, to %s", str(old_chat), str(new_chat)) for mod in MIGRATEABLE: mod.__migrate__(old_chat, new_chat) LOGGER.info("Successfully migrated!") raise DispatcherHandlerStop
def setlog(update: Update, context: CallbackContext): bot = context.bot message = update.effective_message chat = update.effective_chat if chat.type == chat.CHANNEL: message.reply_text( "Ahora, reenvíe /setlog al grupo al que desea vincular este canal!" ) elif message.forward_from_chat: sql.set_chat_log_channel(chat.id, message.forward_from_chat.id) try: message.delete() except BadRequest as excp: if excp.message == "Message to delete not found": pass else: LOGGER.exception( "Error al eliminar el mensaje en el canal de registro. Aunque debería funcionar de todos modos." ) try: bot.send_message( message.forward_from_chat.id, f"Este canal se ha configurado como canal de registro para {chat.title or chat.first_name}.", ) except Unauthorized as excp: if excp.message == "Forbidden: bot is not a member of the channel chat": bot.send_message(chat.id, "Canal log establecido correctamente!") else: LOGGER.exception("ERROR al configurar el canal LOGGER.") bot.send_message(chat.id, "Canal log establecido correctamente!") else: message.reply_text( "Los pasos para configurar un canal de registro son:\n" " - Agregar bot al canal deseado\n" " - Enviar /setlog en el canal\n" " - Reenviar el /setlog al grupo\n")
def broadcast(update: Update, context: CallbackContext): to_send = update.effective_message.text.split(None, 1) if len(to_send) >= 2: to_group = False to_user = False if to_send[0] == "/broadcastgroups": to_group = True if to_send[0] == "/broadcastusers": to_user = True else: to_group = to_user = True chats = sql.get_all_chats() or [] users = get_all_users() failed = 0 failed_user = 0 if to_group: for chat in chats: try: context.bot.sendMessage(int(chat.chat_id), to_send[1]) sleep(0.1) except TelegramError: failed += 1 LOGGER.warning( "Couldn't send broadcast to %s, group name %s", str(chat.chat_id), str(chat.chat_name), ) if to_user: for user in users: try: context.bot.sendMessage(int(user.user_id), to_send[1]) sleep(0.1) except TelegramError: failed_user += 1 LOGGER.warning("Couldn't send broadcast to %s", str(user.user_id)) update.effective_message.reply_text( f"Transmisión completa. {failed} grupos no pudieron recibir el mensaje, probablemente debido a que fui expulsada. {failed_user} no pudieron recibir el mensaje, probablemente debido a que estaba bloqueada" )
def send_log(context: CallbackContext, log_chat_id: str, orig_chat_id: str, result: str): bot = context.bot try: bot.send_message( log_chat_id, result, parse_mode=ParseMode.HTML, disable_web_page_preview=True, ) except BadRequest as excp: if excp.message == "Chat not found": bot.send_message( orig_chat_id, "Este canal de registro ha sido eliminado - desarmando.", ) sql.stop_chat_logging(orig_chat_id) else: LOGGER.warning(excp.message) LOGGER.warning(result) LOGGER.exception("Could not parse") bot.send_message( log_chat_id, result + "\n\nEl formateo se ha deshabilitado debido a un error inesperado.", )
def log_input(update): user = update.effective_user.id chat = update.effective_chat.id LOGGER.info(f"IN: {update.effective_message.text} (user={user}, chat={chat})")
if isfile(f) and f.endswith(".py") and not f.endswith("__init__.py") ] if LOAD or NO_LOAD: to_load = LOAD if to_load: if not all( any(mod == module_name for module_name in all_modules) for mod in to_load): LOGGER.error("Invalid loadorder names. Quitting.") quit(1) all_modules = sorted(set(all_modules) - set(to_load)) to_load = list(all_modules) + to_load else: to_load = all_modules if NO_LOAD: LOGGER.info("Not loading: {}".format(NO_LOAD)) return [item for item in to_load if item not in NO_LOAD] return to_load return all_modules ALL_MODULES = __list_all_modules() LOGGER.info("Modules to load: %s", str(ALL_MODULES)) __all__ = ALL_MODULES + ["ALL_MODULES"]
def settings_button(update: Update, context: CallbackContext): query = update.callback_query user = update.effective_user bot = context.bot mod_match = re.match(r"stngs_module\((.+?),(.+?)\)", query.data) prev_match = re.match(r"stngs_prev\((.+?),(.+?)\)", query.data) next_match = re.match(r"stngs_next\((.+?),(.+?)\)", query.data) back_match = re.match(r"stngs_back\((.+?)\)", query.data) try: if mod_match: chat_id = mod_match.group(1) module = mod_match.group(2) chat = bot.get_chat(chat_id) text = ( "*{}* tiene la siguiente configuración para el módulo *{}*:\n\n" .format(escape_markdown(chat.title), CHAT_SETTINGS[module].__mod_name__) + CHAT_SETTINGS[module].__chat_settings__(chat_id, user.id)) query.message.reply_text( text=text, parse_mode=ParseMode.MARKDOWN, reply_markup=InlineKeyboardMarkup([[ InlineKeyboardButton( text="Atrás", callback_data="stngs_back({})".format(chat_id), ) ]]), ) elif prev_match: chat_id = prev_match.group(1) curr_page = int(prev_match.group(2)) chat = bot.get_chat(chat_id) query.message.reply_text( "Hola! Hay algunas configuraciones para {}. Continúa y elije " "lo que quieras.".format(chat.title), reply_markup=InlineKeyboardMarkup( paginate_modules(curr_page - 1, CHAT_SETTINGS, "stngs", chat=chat_id)), ) elif next_match: chat_id = next_match.group(1) next_page = int(next_match.group(2)) chat = bot.get_chat(chat_id) query.message.reply_text( "¡Hola! Hay algunas configuraciones para {}. Continúa y elije " "lo que te guste.".format(chat.title), reply_markup=InlineKeyboardMarkup( paginate_modules(next_page + 1, CHAT_SETTINGS, "stngs", chat=chat_id)), ) elif back_match: chat_id = back_match.group(1) chat = bot.get_chat(chat_id) query.message.reply_text( text= "¡Hola! Hay algunas configuraciones para {}. Continúa y elije " "lo que te guste.".format(escape_markdown(chat.title)), parse_mode=ParseMode.MARKDOWN, reply_markup=InlineKeyboardMarkup( paginate_modules(0, CHAT_SETTINGS, "stngs", chat=chat_id)), ) # ensure no spinny white circle bot.answer_callback_query(query.id) query.message.delete() except BadRequest as excp: if excp.message == "Message is not modified": pass elif excp.message == "Query_id_invalid": pass elif excp.message == "Message can't be deleted": pass else: LOGGER.exception("Exception in settings buttons. %s", str(query.data))
def reply_filter(update, context): chat = update.effective_chat # type: Optional[Chat] message = update.effective_message # type: Optional[Message] if not update.effective_user or update.effective_user.id == 777000: return to_match = extract_text(message) if not to_match: return chat_filters = sql.get_chat_triggers(chat.id) for keyword in chat_filters: pattern = r"( |^|[^\w])" + re.escape(keyword) + r"( |$|[^\w])" if re.search(pattern, to_match, flags=re.IGNORECASE): filt = sql.get_filter(chat.id, keyword) if filt.reply == "there is should be a new reply": buttons = sql.get_buttons(chat.id, filt.keyword) keyb = build_keyboard_parser(context.bot, chat.id, buttons) keyboard = InlineKeyboardMarkup(keyb) VALID_WELCOME_FORMATTERS = [ "first", "last", "fullname", "username", "id", "chatname", "mention", "user1", "user2", ] if filt.reply_text: valid_format = escape_invalid_curly_brackets( filt.reply_text, VALID_WELCOME_FORMATTERS) if valid_format: filtext = valid_format.format( first=escape(message.from_user.first_name), last=escape(message.from_user.last_name or message.from_user.first_name), fullname=" ".join( [ escape(message.from_user.first_name), escape(message.from_user.last_name), ] if message.from_user.last_name else [escape(message.from_user.first_name)]), username="******" + escape(message.from_user.username) if message.from_user.username else mention_html( message.from_user.id, message.from_user.first_name), mention=mention_html(message.from_user.id, message.from_user.first_name), chatname=escape(message.chat.title) if message.chat.type != "private" else escape( message.from_user.first_name), id=message.from_user.id, ) else: filtext = "" else: filtext = "" if filt.file_type in (sql.Types.BUTTON_TEXT, sql.Types.TEXT): try: context.bot.send_message( chat.id, markdown_to_html(filtext), reply_to_message_id=message.message_id, parse_mode=ParseMode.HTML, disable_web_page_preview=True, reply_markup=keyboard, ) except BadRequest as excp: error_catch = get_exception(excp, filt, chat) if error_catch == "noreply": try: context.bot.send_message( chat.id, markdown_to_html(filtext), parse_mode=ParseMode.HTML, disable_web_page_preview=True, reply_markup=keyboard, ) except BadRequest as excp: LOGGER.exception("Error in filters: " + excp.message) send_message( update.effective_message, get_exception(excp, filt, chat), ) else: try: send_message( update.effective_message, get_exception(excp, filt, chat), ) except BadRequest as excp: LOGGER.exception("Failed to send message: " + excp.message) pass else: if ENUM_FUNC_MAP[ filt.file_type] == dispatcher.bot.send_sticker: ENUM_FUNC_MAP[filt.file_type]( chat.id, filt.file_id, reply_to_message_id=message.message_id, reply_markup=keyboard, ) else: ENUM_FUNC_MAP[filt.file_type]( chat.id, filt.file_id, caption=markdown_to_html(filtext), reply_to_message_id=message.message_id, parse_mode=ParseMode.HTML, reply_markup=keyboard, ) break else: if filt.is_sticker: message.reply_sticker(filt.reply) elif filt.is_document: message.reply_document(filt.reply) elif filt.is_image: message.reply_photo(filt.reply) elif filt.is_audio: message.reply_audio(filt.reply) elif filt.is_voice: message.reply_voice(filt.reply) elif filt.is_video: message.reply_video(filt.reply) elif filt.has_markdown: buttons = sql.get_buttons(chat.id, filt.keyword) keyb = build_keyboard_parser(context.bot, chat.id, buttons) keyboard = InlineKeyboardMarkup(keyb) try: send_message( update.effective_message, filt.reply, parse_mode=ParseMode.MARKDOWN, disable_web_page_preview=True, reply_markup=keyboard, ) except BadRequest as excp: if excp.message == "Protocolo de URL no admitido": try: send_message( update.effective_message, "Parece que intentas utilizar un protocolo de URL no compatible. Telegram " "no admite botones para algunos protocolos, como tg://. Por favor, inténtalo " f"de nuevo, o pregunta en {SUPPORT_CHAT} por ayuda.", ) except BadRequest as excp: LOGGER.exception("Error in filters: " + excp.message) pass elif excp.message == "Reply message not found": try: context.bot.send_message( chat.id, filt.reply, parse_mode=ParseMode.MARKDOWN, disable_web_page_preview=True, reply_markup=keyboard, ) except BadRequest as excp: LOGGER.exception("Error in filters: " + excp.message) pass else: try: send_message( update.effective_message, "Este mensaje no se pudo enviar porque tiene un formato incorrecto.", ) except BadRequest as excp: LOGGER.exception("Error in filters: " + excp.message) pass LOGGER.warning("Message %s could not be parsed", str(filt.reply)) LOGGER.exception( "Could not parse filter %s in chat %s", str(filt.keyword), str(chat.id), ) else: # LEGACY - all new filters will have has_markdown set to True. try: send_message(update.effective_message, filt.reply) except BadRequest as excp: LOGGER.exception("Error in filters: " + excp.message) pass break
def ban(update: Update, context: CallbackContext) -> str: chat = update.effective_chat user = update.effective_user message = update.effective_message log_message = "" bot = context.bot args = context.args user_id, reason = extract_user_and_text(message, args) if not user_id: message.reply_text("Dudo que sea un usuario.") return log_message try: member = chat.get_member(user_id) except BadRequest as excp: if excp.message == "Usuario no encontrado": message.reply_text("Parece que no puedo encontrar a esta persona.") return log_message else: raise if user_id == bot.id: message.reply_text("Oh sí, banéame, noob!") return log_message if is_user_ban_protected(chat, user_id, member) and user not in DEV_USERS: if user_id == OWNER_ID: message.reply_text( "Tratando de ponerme en contra de mi creador, eh?") return log_message elif user_id in DEV_USERS: message.reply_text("¿Estás loco?") return log_message elif user_id in SUDO_USERS: message.reply_text( "Usaría toda la magia existente para hacerlo volar, pero no valdría la pena..." ) return log_message elif user_id in SUPPORT_USERS: message.reply_text( "Lo haría explotar pero es mi soporte ¿Qué hago si me quedo sin magia?" ) return log_message elif user_id in FROG_USERS: message.reply_text( "Lo haría explotar pero es inmune a las explosiones ¿De qué estará hecho?" ) return log_message elif user_id in WHITELIST_USERS: message.reply_text( "Lo haría explotar pero es inmune a las explosiones ¿De qué estará hecho?" ) return log_message else: message.reply_text("Este usuario es inmune, no puedo banearlo..") return log_message log = ( f"<b>{html.escape(chat.title)}:</b>\n" f"#Baneado\n" f"<b>Administrador:</b> {mention_html(user.id, user.first_name)}\n" f"<b>Usuario:</b> {mention_html(member.user.id, member.user.first_name)}" ) if reason: log += "\n<b>Razón:</b> {}".format(reason) try: chat.kick_member(user_id) # bot.send_sticker(chat.id, BAN_STICKER) # banhammer marie sticker bot.sendMessage( chat.id, f"{mention_html(member.user.id, member.user.first_name)} [<code>{member.user.id}</code>] Baneado.", parse_mode=ParseMode.HTML, ) return log except BadRequest as excp: if excp.message == "Mensaje de respuesta no funciona": # Do not reply message.reply_text("Baneado!", quote=False) return log else: LOGGER.warning(update) LOGGER.exception( "ERROR baneando al usuario %s en el chat %s (% s) debido a %s", user_id, chat.title, chat.id, excp.message, ) message.reply_text("Uhm ... eso no funcionó...") return log_message
def temp_ban(update: Update, context: CallbackContext) -> str: chat = update.effective_chat user = update.effective_user message = update.effective_message log_message = "" bot, args = context.bot, context.args user_id, reason = extract_user_and_text(message, args) if not user_id: message.reply_text("Dudo que sea un usuario.") return log_message try: member = chat.get_member(user_id) except BadRequest as excp: if excp.message == "Usuario no encontrado": message.reply_text("Parece que no puedo encontrar a este usuario.") return log_message else: raise if user_id == bot.id: message.reply_text("Yo no voy a BANEAR ¿¿Estás loco??") return log_message if is_user_ban_protected(chat, user_id, member): message.reply_text("No me apetece.") return log_message if not reason: message.reply_text( "No has especificado el tiempo para banear a este usuario!") return log_message split_reason = reason.split(None, 1) time_val = split_reason[0].lower() if len(split_reason) > 1: reason = split_reason[1] else: reason = "" bantime = extract_time(message, time_val) if not bantime: return log_message log = ( f"<b>{html.escape(chat.title)}:</b>\n" "#BaneoTemporal\n" f"<b>Administrador:</b> {mention_html(user.id, user.first_name)}\n" f"<b>Usuario:</b> {mention_html(member.user.id, member.user.first_name)}\n" f"<b>Tiempo:</b> {time_val}") if reason: log += "\n<b>Razón:</b> {}".format(reason) try: chat.kick_member(user_id, until_date=bantime) # bot.send_sticker(chat.id, BAN_STICKER) # banhammer marie sticker bot.sendMessage( chat.id, f"Baneado! {mention_html(member.user.id, member.user.first_name)} " f"será baneado por {time_val}.", parse_mode=ParseMode.HTML, ) return log except BadRequest as excp: if excp.message == "Mensaje de respuesta no encontrado": # Do not reply message.reply_text( f"Baneado! El usuario será baneado por {time_val}.", quote=False) return log else: LOGGER.warning(update) LOGGER.exception( "ERROR baneando al usuario %s en el chat %s (% s) debido a %s", user_id, chat.title, chat.id, excp.message, ) message.reply_text("No puedo banear a ese usuario.") return log_message
def temp_mute(update: Update, context: CallbackContext) -> str: bot, args = context.bot, context.args chat = update.effective_chat user = update.effective_user message = update.effective_message user_id, reason = extract_user_and_text(message, args) reply = check_user(user_id, bot, chat) if reply: message.reply_text(reply) return "" member = chat.get_member(user_id) if not reason: message.reply_text( "No has especificado un tiempo para silenciar a este usuario.!") return "" split_reason = reason.split(None, 1) time_val = split_reason[0].lower() if len(split_reason) > 1: reason = split_reason[1] else: reason = "" mutetime = extract_time(message, time_val) if not mutetime: return "" log = ( f"<b>{html.escape(chat.title)}:</b>\n" f"#SilenciadoTemporal\n" f"<b>Administrador:</b> {mention_html(user.id, user.first_name)}\n" f"<b>Usuario:</b> {mention_html(member.user.id, member.user.first_name)}\n" f"<b>Tiemppo:</b> {time_val}") if reason: log += f"\n<b>Razón:</b> {reason}" try: if member.can_send_messages is None or member.can_send_messages: chat_permissions = ChatPermissions(can_send_messages=False) bot.restrict_chat_member(chat.id, user_id, chat_permissions, until_date=mutetime) bot.sendMessage( chat.id, f"<b>{html.escape(member.user.first_name)}</b> silenciado por {time_val}!", parse_mode=ParseMode.HTML, ) return log else: message.reply_text("Este usuario ya está silenciado.") except BadRequest as excp: if excp.message == "Reply message not found": # Do not reply message.reply_text(f"Silenciado por {time_val}!", quote=False) return log else: LOGGER.warning(update) LOGGER.exception( "ERROR muting user %s in chat %s (%s) due to %s", user_id, chat.title, chat.id, excp.message, ) message.reply_text( "Bueno maldita sea, no puedo silenciar a ese usuario.") return ""
def extract_unt_fedban(message: Message, args: List[str]) -> (Optional[int], Optional[str]): prev_message = message.reply_to_message split_text = message.text.split(None, 1) if len(split_text) < 2: return id_from_reply(message) # only option possible text_to_parse = split_text[1] text = "" entities = list(message.parse_entities([MessageEntity.TEXT_MENTION])) if len(entities) > 0: ent = entities[0] else: ent = None # if entity offset matches (command end/text start) then all good if entities and ent and ent.offset == len( message.text) - len(text_to_parse): ent = entities[0] user_id = ent.user.id text = message.text[ent.offset + ent.length:] elif len(args) >= 1 and args[0][0] == "@": user = args[0] user_id = get_user_id(user) if not user_id and not isinstance(user_id, int): message.reply_text( "No tengo ningún usuario en mi base de datos. Podré interactuar con él si " "Respondes al mensaje de esa persona o me reenvías uno de los mensajes de ese usuario." ) return None, None else: user_id = user_id res = message.text.split(None, 2) if len(res) >= 3: text = res[2] elif len(args) >= 1 and args[0].isdigit(): user_id = int(args[0]) res = message.text.split(None, 2) if len(res) >= 3: text = res[2] elif prev_message: user_id, text = id_from_reply(message) else: return None, None try: message.bot.get_chat(user_id) except BadRequest as excp: if excp.message in ("User_id_invalid", "Chat not found") and not isinstance(user_id, int): message.reply_text( "Parece que no he interactuado con este usuario antes. Reenvíame un mensaje de" "él para darme el control! (como un muñeco vudú, necesito un pedazo de ellos " "para ejecutar ciertos comandos...)") return None, None elif excp.message != "Chat not found": LOGGER.exception("Exception %s on user %s", excp.message, user_id) return None, None elif not isinstance(user_id, int): return None, None return user_id, text
async def process(msg, user, client, reply, replied=None): if not os.path.isdir("resources"): os.mkdir("resources", 0o755) urllib.request.urlretrieve( "https://github.com/erenmetesar/modules-repo/raw/master/Roboto-Regular.ttf", "resources/Roboto-Regular.ttf", ) urllib.request.urlretrieve( "https://github.com/erenmetesar/modules-repo/raw/master/Quivira.otf", "resources/Quivira.otf", ) urllib.request.urlretrieve( "https://github.com/erenmetesar/modules-repo/raw/master/Roboto-Medium.ttf", "resources/Roboto-Medium.ttf", ) urllib.request.urlretrieve( "https://github.com/erenmetesar/modules-repo/raw/master/DroidSansMono.ttf", "resources/DroidSansMono.ttf", ) urllib.request.urlretrieve( "https://github.com/erenmetesar/modules-repo/raw/master/Roboto-Italic.ttf", "resources/Roboto-Italic.ttf", ) # Importıng fonts and gettings the size of text font = ImageFont.truetype("resources/Roboto-Medium.ttf", 43, encoding="utf-16") font2 = ImageFont.truetype("resources/Roboto-Regular.ttf", 33, encoding="utf-16") mono = ImageFont.truetype("resources/DroidSansMono.ttf", 30, encoding="utf-16") italic = ImageFont.truetype("resources/Roboto-Italic.ttf", 33, encoding="utf-16") fallback = ImageFont.truetype("resources/Quivira.otf", 43, encoding="utf-16") # Splitting text maxlength = 0 width = 0 text = [] for line in msg.split("\n"): length = len(line) if length > 43: text += textwrap.wrap(line, 43) maxlength = 43 if width < fallback.getsize(line[:43])[0]: if "MessageEntityCode" in str(reply.entities): width = mono.getsize(line[:43])[0] + 30 else: width = fallback.getsize(line[:43])[0] next else: text.append(line + "\n") if width < fallback.getsize(line)[0]: if "MessageEntityCode" in str(reply.entities): width = mono.getsize(line)[0] + 30 else: width = fallback.getsize(line)[0] if maxlength < length: maxlength = length title = "" try: details = await client( functions.channels.GetParticipantRequest(reply.chat_id, user.id)) if isinstance(details.participant, types.ChannelParticipantCreator): title = (details.participant.rank if details.participant.rank else "Propietario") elif isinstance(details.participant, types.ChannelParticipantAdmin): title = (details.participant.rank if details.participant.rank else "Administrador") except TypeError: pass titlewidth = font2.getsize(title)[0] # Get user name lname = "" if not user.last_name else user.last_name tot = user.first_name + " " + lname namewidth = fallback.getsize(tot)[0] + 10 if namewidth > width: width = namewidth width += titlewidth + 30 if titlewidth > width - namewidth else -( titlewidth - 30) width += 50 height = len(text) * 40 # Profile Photo BG pfpbg = Image.new("RGBA", (125, 600), (0, 0, 0, 0)) # Draw Template top, middle, bottom = await drawer(width, height) # Profile Photo Check and Fetch yes = False color = random.choice(COLORS) async for photo in client.iter_profile_photos(user, limit=1): yes = True if yes: pfp = await client.download_profile_photo(user) paste = Image.open(pfp) try: os.remove(pfp) except Exception as e: LOGGER.exception(f"Error al remover foto de perfil: {e}") paste.thumbnail((105, 105)) # Mask mask_im = Image.new("L", paste.size, 0) draw = ImageDraw.Draw(mask_im) draw.ellipse((0, 0, 105, 105), fill=255) # Apply Mask pfpbg.paste(paste, (0, 0), mask_im) else: paste, color = await no_photo(user, tot) pfpbg.paste(paste, (0, 0)) # Creating a big canvas to gather all the elements canvassize = ( middle.width + pfpbg.width, top.height + middle.height + bottom.height, ) canvas = Image.new("RGBA", canvassize) draw = ImageDraw.Draw(canvas) y = 80 if replied: # Creating a big canvas to gather all the elements replname = "" if not replied.sender.last_name else replied.sender.last_name reptot = replied.sender.first_name + " " + replname replywidth = font2.getsize(reptot)[0] if reply.sticker: sticker = await reply.download_media() stimg = Image.open(sticker) canvas = canvas.resize( (stimg.width + pfpbg.width, stimg.height + 160)) top = Image.new("RGBA", (200 + stimg.width, 300), (29, 29, 29, 255)) draw = ImageDraw.Draw(top) await replied_user(draw, reptot, replied.message.replace("\n", " "), 20) top = top.crop((135, 70, top.width, 300)) canvas.paste(pfpbg, (0, 0)) canvas.paste(top, (pfpbg.width + 10, 0)) canvas.paste(stimg, (pfpbg.width + 10, 140)) os.remove(sticker) return True, canvas canvas = canvas.resize((canvas.width + 60, canvas.height + 120)) top, middle, bottom = await drawer(middle.width + 60, height + 105) canvas.paste(pfpbg, (0, 0)) canvas.paste(top, (pfpbg.width, 0)) canvas.paste(middle, (pfpbg.width, top.height)) canvas.paste(bottom, (pfpbg.width, top.height + middle.height)) draw = ImageDraw.Draw(canvas) if replied.sticker: replied.text = "Sticker" elif replied.photo: replied.text = "Foto" elif replied.audio: replied.text = "Audio" elif replied.voice: replied.text = "Mensaje de voz" elif replied.document: replied.text = "Documento" await replied_user( draw, reptot, replied.message.replace("\n", " "), maxlength + len(title), len(title), ) y = 200 elif reply.sticker: sticker = await reply.download_media() stimg = Image.open(sticker) canvas = canvas.resize( (stimg.width + pfpbg.width + 30, stimg.height + 10)) canvas.paste(pfpbg, (0, 0)) canvas.paste(stimg, (pfpbg.width + 10, 10)) os.remove(sticker) return True, canvas elif reply.document and not reply.audio and not reply.audio: docname = ".".join( reply.document.attributes[-1].file_name.split(".")[:-1]) doctype = reply.document.attributes[-1].file_name.split( ".")[-1].upper() if reply.document.size < 1024: docsize = str(reply.document.size) + " Bytes" elif reply.document.size < 1048576: docsize = str(round(reply.document.size / 1024, 2)) + " KB " elif reply.document.size < 1073741824: docsize = str(round(reply.document.size / 1024**2, 2)) + " MB " else: docsize = str(round(reply.document.size / 1024**3, 2)) + " GB " docbglen = (font.getsize(docsize)[0] if font.getsize(docsize)[0] > font.getsize(docname)[0] else font.getsize(docname)[0]) canvas = canvas.resize((pfpbg.width + width + docbglen, 160 + height)) top, middle, bottom = await drawer(width + docbglen, height + 30) canvas.paste(pfpbg, (0, 0)) canvas.paste(top, (pfpbg.width, 0)) canvas.paste(middle, (pfpbg.width, top.height)) canvas.paste(bottom, (pfpbg.width, top.height + middle.height)) canvas = await doctype(docname, docsize, doctype, canvas) y = 80 if text else 0 else: canvas.paste(pfpbg, (0, 0)) canvas.paste(top, (pfpbg.width, 0)) canvas.paste(middle, (pfpbg.width, top.height)) canvas.paste(bottom, (pfpbg.width, top.height + middle.height)) y = 85 # Writing User's Name space = pfpbg.width + 30 namefallback = ImageFont.truetype("resources/Quivira.otf", 43, encoding="utf-16") for letter in tot: if letter in emoji.UNICODE_EMOJI: newemoji, mask = await emoji_fetch(letter) canvas.paste(newemoji, (space, 24), mask) space += 40 else: if not await fontTest(letter): draw.text((space, 20), letter, font=namefallback, fill=color) space += namefallback.getsize(letter)[0] else: draw.text((space, 20), letter, font=font, fill=color) space += font.getsize(letter)[0] if title: draw.text((canvas.width - titlewidth - 20, 25), title, font=font2, fill="#898989") # Writing all separating emojis and regular texts x = pfpbg.width + 30 bold, mono, italic, link = await get_entity(reply) mdlength = 0 index = 0 emojicount = 0 textfallback = ImageFont.truetype("resources/Quivira.otf", 33, encoding="utf-16") textcolor = "white" for line in text: for letter in line: index = (msg.find(letter) if emojicount == 0 else msg.find(letter) + emojicount) for offset, length in bold.items(): if index in range(offset, length): font2 = ImageFont.truetype("resources/Roboto-Medium.ttf", 33, encoding="utf-16") textcolor = "white" for offset, length in italic.items(): if index in range(offset, length): font2 = ImageFont.truetype("resources/Roboto-Italic.ttf", 33, encoding="utf-16") textcolor = "white" for offset, length in mono.items(): if index in range(offset, length): font2 = ImageFont.truetype("resources/DroidSansMono.ttf", 30, encoding="utf-16") textcolor = "white" for offset, length in link.items(): if index in range(offset, length): font2 = ImageFont.truetype("resources/Roboto-Regular.ttf", 30, encoding="utf-16") textcolor = "#898989" if letter in emoji.UNICODE_EMOJI: newemoji, mask = await emoji_fetch(letter) canvas.paste(newemoji, (x, y - 2), mask) x += 45 emojicount += 1 else: if not await fontTest(letter): draw.text((x, y), letter, font=textfallback, fill=textcolor) x += textfallback.getsize(letter)[0] else: draw.text((x, y), letter, font=font2, fill=textcolor) x += font2.getsize(letter)[0] msg = msg.replace(letter, "¶", 1) y += 40 x = pfpbg.width + 30 return True, canvas
def report(update: Update, context: CallbackContext) -> str: bot = context.bot args = context.args message = update.effective_message chat = update.effective_chat user = update.effective_user if chat and message.reply_to_message and sql.chat_should_report(chat.id): reported_user = message.reply_to_message.from_user chat_name = chat.title or chat.first or chat.username admin_list = chat.get_administrators() message = update.effective_message if not args: message.reply_text("Agrega una razón para reportar primero.") return "" if user.id == reported_user.id: message.reply_text("Uh sí, seguro seguro... mucho más?") return "" if user.id == bot.id: message.reply_text("Buen intento.") return "" if reported_user.id in REPORT_IMMUNE_USERS: message.reply_text( "Eh? Estás reportando a los usuarios de la lista blanca?" ) return "" if chat.username and chat.type == Chat.SUPERGROUP: reported = f"{mention_html(user.id, user.first_name)} reportado {mention_html(reported_user.id, reported_user.first_name)} a los admins!" msg = ( f"<b>⚠️ Reporte: </b>{html.escape(chat.title)}\n" f"<b> • Reportado por:</b> {mention_html(user.id, user.first_name)}(<code>{user.id}</code>)\n" f"<b> • Usuario reportado:</b> {mention_html(reported_user.id, reported_user.first_name)} (<code>{reported_user.id}</code>)\n" ) link = f'<b> • Mensaje reportado:</b> <a href="https://t.me/{chat.username}/{message.reply_to_message.message_id}">Click aquí</a>' should_forward = False keyboard = [ [ InlineKeyboardButton( "➡ Mensaje", url=f"https://t.me/{chat.username}/{message.reply_to_message.message_id}", ) ], [ InlineKeyboardButton( "⚠ Kickear", callback_data=f"report_{chat.id}=kick={reported_user.id}={reported_user.first_name}", ), InlineKeyboardButton( "⛔️ Banear", callback_data=f"report_{chat.id}=banned={reported_user.id}={reported_user.first_name}", ), ], [ InlineKeyboardButton( "❎ Eliminar Mensaje", callback_data=f"report_{chat.id}=delete={reported_user.id}={message.reply_to_message.message_id}", ) ], ] reply_markup = InlineKeyboardMarkup(keyboard) else: reported = ( f"{mention_html(user.id, user.first_name)} reportado " f"{mention_html(reported_user.id, reported_user.first_name)} a los admins!" ) msg = f'{mention_html(user.id, user.first_name)} está llamando a los administradores en "{html.escape(chat_name)}"!' link = "" should_forward = True for admin in admin_list: if admin.user.is_bot: # can't message bots continue if sql.user_should_report(admin.user.id): try: if not chat.type == Chat.SUPERGROUP: bot.send_message( admin.user.id, msg + link, parse_mode=ParseMode.HTML ) if should_forward: message.reply_to_message.forward(admin.user.id) if ( len(message.text.split()) > 1 ): # If user is giving a reason, send his message too message.forward(admin.user.id) if not chat.username: bot.send_message( admin.user.id, msg + link, parse_mode=ParseMode.HTML ) if should_forward: message.reply_to_message.forward(admin.user.id) if ( len(message.text.split()) > 1 ): # If user is giving a reason, send his message too message.forward(admin.user.id) if chat.username and chat.type == Chat.SUPERGROUP: bot.send_message( admin.user.id, msg + link, parse_mode=ParseMode.HTML, reply_markup=reply_markup, ) if should_forward: message.reply_to_message.forward(admin.user.id) if ( len(message.text.split()) > 1 ): # If user is giving a reason, send his message too message.forward(admin.user.id) except Unauthorized: pass except BadRequest as excp: # TODO: cleanup exceptions LOGGER.exception("Exception while reporting user") message.reply_to_message.reply_text( f"{mention_html(user.id, user.first_name)} mensaje reportado a los admins.", parse_mode=ParseMode.HTML, ) return msg return ""
if WEBHOOK: LOGGER.info("Usando webhooks.") updater.start_webhook(listen="0.0.0.0", port=PORT, url_path=TOKEN) if CERT_PATH: updater.bot.set_webhook(url=URL + TOKEN, certificate=open(CERT_PATH, "rb")) else: updater.bot.set_webhook(url=URL + TOKEN) else: LOGGER.info("Usando long polling.") updater.start_polling(timeout=15, read_latency=4, drop_pending_updates=True) if len(argv) not in (1, 3, 4): telethn.disconnect() else: telethn.run_until_disconnected() updater.idle() if __name__ == "__main__": LOGGER.info("Módulos cargados correctamente: " + str(ALL_MODULES)) telethn.start(bot_token=TOKEN) pyrogrm.start() main() idle()
def get(update, context, notename, show_none=True, no_format=False): bot = context.bot chat_id = update.effective_chat.id note = sql.get_note(chat_id, notename) message = update.effective_message # type: Optional[Message] if note: if MessageHandlerChecker.check_user(update.effective_user.id): return # If we're replying to a message, reply to that message (unless it's an error) if message.reply_to_message: reply_id = message.reply_to_message.message_id else: reply_id = message.message_id if note.is_reply: if JOIN_LOGGER: try: bot.forward_message(chat_id=chat_id, from_chat_id=JOIN_LOGGER, message_id=note.value) except BadRequest as excp: if excp.message == "Reply message not found": message.reply_text( "Parece que este mensaje se ha perdido; lo eliminaré " "de la lista de notas.") sql.rm_note(chat_id, notename) else: raise else: try: bot.forward_message(chat_id=chat_id, from_chat_id=chat_id, message_id=note.value) except BadRequest as excp: if excp.message == "Message to forward not found": message.reply_text( "Parece que se ha eliminado el remitente original de esta nota " "su mensaje - ¡lo siento! Haga que su administrador de bot comience a usar un " "guardado de mensajes para evitar esto. Eliminaré esta nota de " "sus notas guardadas.") sql.rm_note(chat_id, notename) else: raise else: VALID_NOTE_FORMATTERS = [ "first", "last", "fullname", "username", "id", "chatname", "mention", ] valid_format = escape_invalid_curly_brackets( note.value, VALID_NOTE_FORMATTERS) if valid_format: text = valid_format.format( first=escape_markdown(message.from_user.first_name), last=escape_markdown(message.from_user.last_name or message.from_user.first_name), fullname=escape_markdown( " ".join([ message.from_user.first_name, message.from_user. last_name ] if message.from_user.last_name else [message.from_user.first_name])), username="******" + message.from_user.username if message.from_user.username else mention_markdown( message.from_user.id, message.from_user.first_name), mention=mention_markdown(message.from_user.id, message.from_user.first_name), chatname=escape_markdown( message.chat.title if message.chat.type != "private" else message.from_user.first_name), id=message.from_user.id, ) else: text = "" keyb = [] parseMode = ParseMode.MARKDOWN buttons = sql.get_buttons(chat_id, notename) if no_format: parseMode = None text += revert_buttons(buttons) else: keyb = build_keyboard(buttons) keyboard = InlineKeyboardMarkup(keyb) try: if note.msgtype in (sql.Types.BUTTON_TEXT, sql.Types.TEXT): bot.send_message( chat_id, text, reply_to_message_id=reply_id, parse_mode=parseMode, disable_web_page_preview=True, reply_markup=keyboard, ) else: ENUM_FUNC_MAP[note.msgtype]( chat_id, note.file, caption=text, reply_to_message_id=reply_id, parse_mode=parseMode, reply_markup=keyboard, ) except BadRequest as excp: if excp.message == "Entity_mention_user_invalid": message.reply_text( "Parece que trataste de mencionar a alguien a quien nunca había visto antes. Si tú realmente " "quieres mencionarlos, reenviarme uno de sus mensajes y podré " "etiquetarlo!") elif FILE_MATCHER.match(note.value): message.reply_text( "Esta nota era un archivo importado incorrectamente de otro bot; no puedo usar " "eso. Si realmente lo necesita, tendrá que guardarlo nuevamente. Mientras" "tanto, lo eliminaré de tu lista de notas.") sql.rm_note(chat_id, notename) else: message.reply_text( "Esta nota no se pudo enviar porque tiene un formato incorrecto. Entra a " f"@{SUPPORT_CHAT} si no puedes entender por qué!") LOGGER.exception("Could not parse message #%s in chat %s", notename, str(chat_id)) LOGGER.warning("Message was: %s", str(note.value)) return elif show_none: message.reply_text("Esta nota no existe")
def send(update, message, keyboard, backup_message): chat = update.effective_chat cleanserv = sql.clean_service(chat.id) reply = update.message.message_id # Clean service welcome if cleanserv: try: dispatcher.bot.delete_message(chat.id, update.message.message_id) except BadRequest: pass reply = False try: msg = update.effective_message.reply_text( message, parse_mode=ParseMode.MARKDOWN, reply_markup=keyboard, reply_to_message_id=reply, ) except BadRequest as excp: if excp.message == "Reply message not found": msg = update.effective_message.reply_text( message, parse_mode=ParseMode.MARKDOWN, reply_markup=keyboard, quote=False, ) elif excp.message == "Button_url_invalid": msg = update.effective_message.reply_text( markdown_parser( backup_message + "\nNota: El mensaje actual tiene una URL no válida " "en uno de sus botones. Por favor actualice."), parse_mode=ParseMode.MARKDOWN, reply_to_message_id=reply, ) elif excp.message == "Unsupported url protocol": msg = update.effective_message.reply_text( markdown_parser( backup_message + "\nNota: el mensaje actual tiene botones que " "utilizan protocolos de URL que no son compatibles con " "Telegram. Por favor actualice."), parse_mode=ParseMode.MARKDOWN, reply_to_message_id=reply, ) elif excp.message == "Wrong url host": msg = update.effective_message.reply_text( markdown_parser( backup_message + "\nNota: el mensaje actual tiene algunas URL incorrectas. " "Por favor actualice."), parse_mode=ParseMode.MARKDOWN, reply_to_message_id=reply, ) LOGGER.warning(message) LOGGER.warning(keyboard) LOGGER.exception( "No se pudo analizar! obtuve errores de host de URL no válidas" ) else: msg = update.effective_message.reply_text( markdown_parser(backup_message + "\nNota: Se produjo un error al enviar el " "mensaje personalizado. Por favor actualice."), parse_mode=ParseMode.MARKDOWN, reply_to_message_id=reply, ) LOGGER.exception() return msg
def import_data(update, context): msg = update.effective_message chat = update.effective_chat user = update.effective_user # TODO: allow uploading doc with command, not just as reply # only work with a doc conn = connected(context.bot, update, chat, user.id, need_admin=True) if conn: chat = dispatcher.bot.getChat(conn) chat_name = dispatcher.bot.getChat(conn).title else: if update.effective_message.chat.type == "private": update.effective_message.reply_text( "Este es un comando solo para grupos!") return "" chat = update.effective_chat chat_name = update.effective_message.chat.title if msg.reply_to_message and msg.reply_to_message.document: try: file_info = context.bot.get_file( msg.reply_to_message.document.file_id) except BadRequest: msg.reply_text( "Intente descargar y cargar el archivo usted mismo nuevamente, este me parece roto!" ) return with BytesIO() as file: file_info.download(out=file) file.seek(0) data = json.load(file) # only import one group if len(data) > 1 and str(chat.id) not in data: msg.reply_text( "Hay más de un grupo en este archivo y el chat.id no es el mismo. ¿Cómo se supone que debo importarlo??" ) return # Check if backup is this chat try: if data.get(str(chat.id)) is None: if conn: text = "La copia de seguridad proviene de otro chat, no puedo devolver otro chat al chat *{}*".format( chat_name) else: text = "La copia de seguridad proviene de otro chat, no puedo devolver otro chat a este chat" return msg.reply_text(text, parse_mode="markdown") except Exception: return msg.reply_text("Hubo un problema al importar los datos!") # Check if backup is from self try: if str(context.bot.id) != str(data[str(chat.id)]["bot"]): return msg.reply_text( "La copia de seguridad de otro bot que no se sugiere puede causar el problema, los documentos, las fotos, los videos, los audios, los registros pueden no funcionar como deberían.." ) except Exception: pass # Select data source if str(chat.id) in data: data = data[str(chat.id)]["hashes"] else: data = data[list(data.keys())[0]]["hashes"] try: for mod in DATA_IMPORT: mod.__import_data__(str(chat.id), data) except Exception: msg.reply_text( f"Se produjo un error al recuperar tus datos. El proceso falló. Si tienes algún problema con esto, llévelo a @{SUPPORT_CHAT}" ) LOGGER.exception( "Falló la importación para el chat %s con el nombre %s.", str(chat.id), str(chat.title), ) return # TODO: some of that link logic # NOTE: consider default permissions stuff? if conn: text = "Copia de seguridad completamente restaurada en *{}*.".format( chat_name) else: text = "Copia de seguridad completamente restaurada." msg.reply_text(text, parse_mode="markdown")
def del_lockables(update, context): chat = update.effective_chat # type: Optional[Chat] message = update.effective_message # type: Optional[Message] for lockable, filter in LOCK_TYPES.items(): if lockable == "rtl": if sql.is_locked(chat.id, lockable) and can_delete( chat, context.bot.id): if message.caption: check = ad.detect_alphabet(u"{}".format(message.caption)) if "ARABIC" in check: try: message.delete() except BadRequest as excp: if excp.message == "Mensaje para eliminar no encontrado": pass else: LOGGER.exception("ERROR en bloqueables") break if message.text: check = ad.detect_alphabet(u"{}".format(message.text)) if "ARABIC" in check: try: message.delete() except BadRequest as excp: if excp.message == "Mensaje para eliminar no encontrado": pass else: LOGGER.exception("ERROR en bloqueables") break continue if lockable == "button": if sql.is_locked(chat.id, lockable) and can_delete( chat, context.bot.id): if message.reply_markup and message.reply_markup.inline_keyboard: try: message.delete() except BadRequest as excp: if excp.message == "Mensaje para eliminar no encontrado": pass else: LOGGER.exception("ERROR en bloqueables") break continue if lockable == "inline": if sql.is_locked(chat.id, lockable) and can_delete( chat, context.bot.id): if message and message.via_bot: try: message.delete() except BadRequest as excp: if excp.message == "Mensaje para eliminar no encontrado": pass else: LOGGER.exception("ERROR en bloqueables") break continue if (filter(update) and sql.is_locked(chat.id, lockable) and can_delete(chat, context.bot.id)): if lockable == "bots": new_members = update.effective_message.new_chat_members for new_mem in new_members: if new_mem.is_bot: if not is_bot_admin(chat, context.bot.id): send_message( update.effective_message, "Veo un bot y me han dicho que evite que se unan..." "Pero no tengo admin! :(", ) return chat.kick_member(new_mem.id) send_message( update.effective_message, "Solo los admins pueden agregar bots en este chat! Fuera de aquí.", ) break else: try: message.delete() except BadRequest as excp: if excp.message == "Mensaje para eliminar no encontrado": pass else: LOGGER.exception("ERROR en bloqueables") break
dispatcher.add_handler(migrate_handler) dispatcher.add_handler(donate_handler) dispatcher.add_error_handler(error_callback) if WEBHOOK: LOGGER.info("Using webhooks.") updater.start_webhook(listen="0.0.0.0", port=PORT, url_path=TOKEN) if CERT_PATH: updater.bot.set_webhook(url=URL + TOKEN, certificate=open(CERT_PATH, "rb")) else: updater.bot.set_webhook(url=URL + TOKEN) else: LOGGER.info("Using long polling.") updater.start_polling(timeout=15, read_latency=4, clean=True) if len(argv) not in (1, 3, 4): telethn.disconnect() else: telethn.run_until_disconnected() updater.idle() if __name__ == "__main__": LOGGER.info("Successfully loaded modules: " + str(ALL_MODULES)) telethn.start(bot_token=TOKEN) pyrogrm.start() main() idle()