2022-04-14 07:18:45 +00:00
|
|
|
|
import datetime
|
2022-05-20 10:23:15 +00:00
|
|
|
|
import time
|
2022-06-22 13:33:07 +00:00
|
|
|
|
from functools import wraps
|
2022-05-20 10:23:15 +00:00
|
|
|
|
from typing import Callable, Optional
|
2022-04-14 07:18:45 +00:00
|
|
|
|
|
2022-05-26 11:51:06 +00:00
|
|
|
|
from telegram import Update, ReplyKeyboardRemove
|
2022-05-20 10:23:15 +00:00
|
|
|
|
from telegram.constants import ChatType
|
2022-04-20 08:32:07 +00:00
|
|
|
|
from telegram.error import BadRequest
|
2022-05-18 14:09:07 +00:00
|
|
|
|
from telegram.ext import CallbackContext, ConversationHandler, filters
|
2022-04-14 07:18:45 +00:00
|
|
|
|
|
2022-04-20 08:32:07 +00:00
|
|
|
|
from logger import Log
|
2022-04-14 07:18:45 +00:00
|
|
|
|
from service import BaseService
|
|
|
|
|
|
|
|
|
|
|
2022-06-03 07:54:56 +00:00
|
|
|
|
async def clean_message(context: CallbackContext, chat_id: int, message_id: int) -> bool:
|
|
|
|
|
try:
|
|
|
|
|
await context.bot.delete_message(chat_id=chat_id, message_id=message_id)
|
|
|
|
|
return True
|
|
|
|
|
except BadRequest as error:
|
|
|
|
|
if "not found" in str(error):
|
|
|
|
|
Log.warning(f"定时删除消息 chat_id[{chat_id}] message_id[{message_id}]失败 消息不存在")
|
|
|
|
|
elif "Message can't be deleted" in str(error):
|
|
|
|
|
Log.warning(f"定时删除消息 chat_id[{chat_id}] message_id[{message_id}]失败 消息无法删除 可能是没有授权")
|
|
|
|
|
else:
|
|
|
|
|
Log.warning(f"定时删除消息 chat_id[{chat_id}] message_id[{message_id}]失败 \n", error)
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def add_delete_message_job(context: CallbackContext, chat_id: int, message_id: int,
|
|
|
|
|
delete_seconds: int = 60):
|
|
|
|
|
context.job_queue.scheduler.add_job(clean_message, "date",
|
|
|
|
|
id=f"{chat_id}|{message_id}|auto_clean_message",
|
|
|
|
|
name=f"{chat_id}|{message_id}|auto_clean_message",
|
|
|
|
|
args=[context, chat_id, message_id],
|
|
|
|
|
run_date=context.job_queue._tz_now() + datetime.timedelta(
|
|
|
|
|
seconds=delete_seconds), replace_existing=True)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class BasePlugins:
|
|
|
|
|
def __init__(self, service: BaseService):
|
|
|
|
|
self.service = service
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
async def cancel(update: Update, _: CallbackContext) -> int:
|
|
|
|
|
await update.message.reply_text("退出命令", reply_markup=ReplyKeyboardRemove())
|
|
|
|
|
return ConversationHandler.END
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
async def _clean(context: CallbackContext, chat_id: int, message_id: int) -> bool:
|
|
|
|
|
return await clean_message(context, chat_id, message_id)
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
def _add_delete_message_job(context: CallbackContext, chat_id: int, message_id: int,
|
|
|
|
|
delete_seconds: int = 60):
|
|
|
|
|
return add_delete_message_job(context, chat_id, message_id, delete_seconds)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class NewChatMembersHandler:
|
2022-06-22 13:33:07 +00:00
|
|
|
|
def __init__(self, service: BaseService, auth_callback: Callable):
|
2022-06-03 07:54:56 +00:00
|
|
|
|
self.service = service
|
|
|
|
|
self.auth_callback = auth_callback
|
|
|
|
|
|
|
|
|
|
async def new_member(self, update: Update, context: CallbackContext) -> None:
|
|
|
|
|
message = update.message
|
|
|
|
|
chat = message.chat
|
|
|
|
|
from_user = message.from_user
|
|
|
|
|
quit_status = False
|
|
|
|
|
if filters.ChatType.GROUPS.filter(message):
|
|
|
|
|
for user in message.new_chat_members:
|
|
|
|
|
if user.id == context.bot.id:
|
|
|
|
|
if from_user is not None:
|
|
|
|
|
Log.info(f"用户 {from_user.full_name}[{from_user.id}] 在群 {chat.title}[{chat.id}] 邀请BOT")
|
|
|
|
|
admin_list = await self.service.admin.get_admin_list()
|
|
|
|
|
if from_user.id in admin_list:
|
|
|
|
|
await context.bot.send_message(message.chat_id,
|
|
|
|
|
'感谢邀请小派蒙到本群!请使用 /help 查看咱已经学会的功能。')
|
|
|
|
|
else:
|
|
|
|
|
quit_status = True
|
|
|
|
|
else:
|
|
|
|
|
Log.info(f"未知用户 在群 {chat.title}[{chat.id}] 邀请BOT")
|
|
|
|
|
quit_status = True
|
|
|
|
|
if quit_status:
|
|
|
|
|
Log.warning("不是管理员邀请!退出群聊。")
|
|
|
|
|
await context.bot.send_message(message.chat_id, "派蒙不想进去!不是旅行者的邀请!")
|
|
|
|
|
await context.bot.leave_chat(chat.id)
|
|
|
|
|
await self.auth_callback(update, context)
|
|
|
|
|
|
|
|
|
|
|
2022-06-26 05:47:50 +00:00
|
|
|
|
def restricts(filters_chat: filters = filters.ALL, return_data=None, try_delete_message: bool = False,
|
2022-06-22 13:33:07 +00:00
|
|
|
|
restricts_time: int = 5):
|
2022-05-20 10:23:15 +00:00
|
|
|
|
"""
|
2022-06-22 13:33:07 +00:00
|
|
|
|
用于装饰在指定函数防止洪水调用的装饰器
|
|
|
|
|
|
|
|
|
|
被修饰的函数传入参数必须为
|
|
|
|
|
async def command_func(update, context)
|
|
|
|
|
或
|
|
|
|
|
async def command_func(self, update, context)
|
|
|
|
|
|
|
|
|
|
如果修饰的函数属于
|
|
|
|
|
ConversationHandler
|
|
|
|
|
参数
|
|
|
|
|
return_data
|
|
|
|
|
必须传入
|
|
|
|
|
ConversationHandler.END
|
|
|
|
|
|
|
|
|
|
我真™是服了某些闲着没事干的群友了
|
2022-05-20 10:23:15 +00:00
|
|
|
|
"""
|
2022-06-22 13:33:07 +00:00
|
|
|
|
|
|
|
|
|
def decorator(func: Callable):
|
|
|
|
|
@wraps(func)
|
|
|
|
|
async def restricts_func(*args, **kwargs):
|
2022-05-20 10:23:15 +00:00
|
|
|
|
update: Optional[Update] = None
|
2022-06-22 13:33:07 +00:00
|
|
|
|
context: Optional[CallbackContext] = None
|
|
|
|
|
if len(args) == 3:
|
|
|
|
|
# self update context
|
|
|
|
|
_, update, context = args
|
|
|
|
|
elif len(args) == 2:
|
|
|
|
|
# update context
|
|
|
|
|
update, context = args
|
|
|
|
|
else:
|
2022-05-20 10:23:15 +00:00
|
|
|
|
return await func(*args, **kwargs)
|
|
|
|
|
message = update.message
|
|
|
|
|
user = update.effective_user
|
2022-06-22 13:33:07 +00:00
|
|
|
|
if filters_chat.filter(message):
|
|
|
|
|
command_time = context.user_data.get("command_time", 0)
|
|
|
|
|
count = context.user_data.get("usage_count", 0)
|
|
|
|
|
restrict_since = context.user_data.get("restrict_since", 0)
|
|
|
|
|
# 洪水防御
|
|
|
|
|
if restrict_since:
|
|
|
|
|
if (time.time() - restrict_since) >= 60 * 5:
|
|
|
|
|
del context.user_data["restrict_since"]
|
|
|
|
|
del context.user_data["usage_count"]
|
2022-05-20 10:23:15 +00:00
|
|
|
|
else:
|
2022-06-22 13:33:07 +00:00
|
|
|
|
return return_data
|
|
|
|
|
else:
|
|
|
|
|
if count == 5:
|
|
|
|
|
context.user_data["restrict_since"] = time.time()
|
2022-06-22 13:57:43 +00:00
|
|
|
|
await update.effective_message.reply_text("你已经触发洪水防御,请等待5分钟")
|
2022-06-22 13:33:07 +00:00
|
|
|
|
Log.warning(f"用户 {user.full_name}[{user.id}] 触发洪水限制 已被限制5分钟")
|
|
|
|
|
return return_data
|
|
|
|
|
# 单次使用限制
|
|
|
|
|
if command_time:
|
|
|
|
|
if (time.time() - command_time) <= restricts_time:
|
|
|
|
|
context.user_data["usage_count"] = count + 1
|
2022-06-22 13:44:40 +00:00
|
|
|
|
if filters.ChatType.GROUPS.filter(message):
|
|
|
|
|
if try_delete_message:
|
|
|
|
|
try:
|
2022-06-22 13:33:07 +00:00
|
|
|
|
await message.delete()
|
2022-06-22 13:44:40 +00:00
|
|
|
|
except BadRequest as error:
|
|
|
|
|
Log.warning("删除消息失败", error)
|
|
|
|
|
return return_data
|
2022-06-22 13:55:27 +00:00
|
|
|
|
else:
|
|
|
|
|
if count >= 1:
|
|
|
|
|
context.user_data["usage_count"] = count - 1
|
2022-06-22 13:33:07 +00:00
|
|
|
|
|
|
|
|
|
context.user_data["command_time"] = time.time()
|
|
|
|
|
|
2022-05-20 10:23:15 +00:00
|
|
|
|
return await func(*args, **kwargs)
|
|
|
|
|
|
2022-06-22 13:33:07 +00:00
|
|
|
|
return restricts_func
|
|
|
|
|
|
|
|
|
|
return decorator
|