zhenxun_bot/basic_plugins/admin_bot_manage/_data_source.py

385 lines
14 KiB
Python
Raw Normal View History

2022-02-19 18:20:19 +08:00
from typing import List
from nonebot.adapters.onebot.v11.message import MessageSegment
from services.log import logger
2022-12-26 18:40:34 +08:00
from configs.path_config import DATA_PATH, IMAGE_PATH
2022-02-19 18:20:19 +08:00
from utils.message_builder import image
2022-02-23 13:51:19 +08:00
from utils.utils import get_bot, get_matchers
2022-02-19 18:20:19 +08:00
from pathlib import Path
from models.group_member_info import GroupInfoUser
from datetime import datetime
from services.db_context import db
from models.level_user import LevelUser
from configs.config import Config
from utils.manager import group_manager, plugins2settings_manager, plugins_manager
from utils.image_utils import BuildImage
from utils.http_utils import AsyncHttpx
import asyncio
import time
import os
import ujson as json
2022-02-19 18:20:19 +08:00
custom_welcome_msg_json = (
Path() / "data" / "custom_welcome_msg" / "custom_welcome_msg.json"
)
2022-02-19 18:20:19 +08:00
2022-12-26 18:40:34 +08:00
ICON_PATH = IMAGE_PATH / 'other'
2022-02-19 18:20:19 +08:00
2022-12-26 18:40:34 +08:00
async def group_current_status(group_id: int) -> str:
2022-02-19 18:20:19 +08:00
"""
说明:
获取当前群聊所有通知的开关
参数:
:param group_id: 群号
2022-02-19 18:20:19 +08:00
"""
_data = group_manager.get_task_data()
2022-12-26 18:40:34 +08:00
image_list = []
for i, task in enumerate(_data):
name = _data[task]
name_image = BuildImage(0, 0, plain_text=f"{i+1}.{name}", font_size=20)
bk = BuildImage(name_image.w + 200, name_image.h + 20, color=(103, 177, 109), font_size=15)
await bk.apaste(name_image, (10, 0), True, "by_height")
a_icon = BuildImage(40, 40, background=ICON_PATH / "btn_false.png")
if group_manager.check_group_task_status(group_id, task):
a_icon = BuildImage(40, 40, background=ICON_PATH / "btn_true.png")
b_icon = BuildImage(40, 40, background=ICON_PATH / "btn_false.png")
if group_manager.check_task_super_status(task):
b_icon = BuildImage(40, 40, background=ICON_PATH / "btn_true.png")
await bk.atext((name_image.w + 20, 10), "状态")
await bk.apaste(a_icon, (name_image.w + 50, 0), True)
await bk.atext((name_image.w + 100, 10), "全局")
await bk.apaste(b_icon, (name_image.w + 130, 0), True)
image_list.append(bk)
w = max([x.w for x in image_list])
h = sum([x.h + 10 for x in image_list])
A = BuildImage(w + 20, h + 70, font_size=30, color=(119, 97, 177))
await A.atext((15, 20), "群被动状态")
curr_h = 75
for img in image_list:
# await img.acircle_corner()
await A.apaste(img, (0, curr_h), True)
curr_h += img.h + 10
return A.pic2bs4()
2022-02-19 18:20:19 +08:00
async def custom_group_welcome(
msg: str, img_list: List[str], user_id: int, group_id: int
2022-02-19 18:20:19 +08:00
) -> str:
"""
说明:
替换群欢迎消息
参数:
:param msg: 欢迎消息文本
:param img_list: 欢迎消息图片只取第一张
:param user_id: 用户id用于log记录
:param group_id: 群号
2022-02-19 18:20:19 +08:00
"""
img_result = ""
result = ""
img = img_list[0] if img_list else ""
2022-02-19 18:20:19 +08:00
if (DATA_PATH / f"custom_welcome_msg/{group_id}.jpg").exists():
(DATA_PATH / f"custom_welcome_msg/{group_id}.jpg").unlink()
data = {}
2022-02-19 18:20:19 +08:00
if not custom_welcome_msg_json.exists():
custom_welcome_msg_json.parent.mkdir(parents=True, exist_ok=True)
else:
try:
data = json.load(open(custom_welcome_msg_json, "r"))
except FileNotFoundError:
pass
2022-02-19 18:20:19 +08:00
try:
if msg:
data[str(group_id)] = str(msg)
json.dump(
data, open(custom_welcome_msg_json, "w"), indent=4, ensure_ascii=False
)
logger.info(f"USER {user_id} GROUP {group_id} 更换群欢迎消息 {msg}")
result += msg
if img:
await AsyncHttpx.download_file(
img, DATA_PATH / "custom_welcome_msg" / f"{group_id}.jpg"
)
img_result = image(DATA_PATH / "custom_welcome_msg" / f"{group_id}.jpg")
2022-02-19 18:20:19 +08:00
logger.info(f"USER {user_id} GROUP {group_id} 更换群欢迎消息图片")
except Exception as e:
logger.error(f"GROUP {group_id} 替换群消息失败 e:{e}")
return "替换群消息失败.."
return f"替换群欢迎消息成功:\n{result}" + img_result
task_data = None
2022-12-26 18:40:34 +08:00
def change_global_task_status(cmd: str) -> str:
"""
说明:
修改全局被动任务状态
参数:
:param cmd: 功能名称
"""
global task_data
if not task_data:
task_data = group_manager.get_task_data()
status = cmd[:2]
_cmd = cmd[4:]
if '全部被动' in cmd:
for task in task_data:
if status == "开启":
group_manager.open_global_task(task)
else:
group_manager.close_global_task(task)
group_manager.save()
return f"{status} 全局全部被动技能!"
else:
modules = [x for x in task_data if task_data[x].lower() == _cmd.lower()]
if not modules:
return '未查询到该被动任务'
if status == "开启":
group_manager.open_global_task(modules[0])
else:
group_manager.close_global_task(modules[0])
group_manager.save()
return f"{status} 全局{_cmd}"
async def change_group_switch(cmd: str, group_id: int, is_super: bool = False) -> str:
2022-02-19 18:20:19 +08:00
"""
说明:
修改群功能状态
参数:
:param cmd: 功能名称
:param group_id: 群号
:param is_super: 是否位超级用户超级用户用于私聊开关功能状态
2022-02-19 18:20:19 +08:00
"""
global task_data
2022-02-19 18:20:19 +08:00
if not task_data:
task_data = group_manager.get_task_data()
group_help_file = DATA_PATH / "group_help" / f"{group_id}.png"
status = cmd[:2]
cmd = cmd[2:]
type_ = "plugin"
modules = plugins2settings_manager.get_plugin_module(cmd, True)
if cmd == "全部被动":
for task in task_data:
if status == "开启":
2022-11-21 20:43:41 +08:00
if not group_manager.check_group_task_status(group_id, task):
group_manager.open_group_task(group_id, task)
2022-02-19 18:20:19 +08:00
else:
2022-11-21 20:43:41 +08:00
if group_manager.check_group_task_status(group_id, task):
group_manager.close_group_task(group_id, task)
2022-02-19 18:20:19 +08:00
if group_help_file.exists():
group_help_file.unlink()
return f"{status} 全部被动技能!"
if cmd == "全部功能":
for f in plugins2settings_manager.get_data():
if status == "开启":
2022-11-21 20:43:41 +08:00
group_manager.unblock_plugin(f, group_id, False)
2022-02-19 18:20:19 +08:00
else:
2022-11-21 20:43:41 +08:00
group_manager.block_plugin(f, group_id, False)
group_manager.save()
if group_help_file.exists():
group_help_file.unlink()
2022-02-19 18:20:19 +08:00
return f"{status} 全部功能!"
if cmd.lower() in [task_data[x].lower() for x in task_data.keys()]:
2022-02-19 18:20:19 +08:00
type_ = "task"
modules = [x for x in task_data.keys() if task_data[x].lower() == cmd.lower()]
2022-02-19 18:20:19 +08:00
for module in modules:
if is_super:
module = f"{module}:super"
if status == "开启":
if type_ == "task":
2022-11-21 20:43:41 +08:00
if group_manager.check_group_task_status(group_id, module):
2022-02-19 18:20:19 +08:00
return f"被动 {task_data[module]} 正处于开启状态!不要重复开启."
2022-11-21 20:43:41 +08:00
group_manager.open_group_task(group_id, module)
2022-02-19 18:20:19 +08:00
else:
if group_manager.get_plugin_status(module, group_id):
return f"功能 {cmd} 正处于开启状态!不要重复开启."
group_manager.unblock_plugin(module, group_id)
else:
if type_ == "task":
2022-11-21 20:43:41 +08:00
if not group_manager.check_group_task_status(group_id, module):
2022-02-19 18:20:19 +08:00
return f"被动 {task_data[module]} 正处于关闭状态!不要重复关闭."
2022-11-21 20:43:41 +08:00
group_manager.close_group_task(group_id, module)
2022-02-19 18:20:19 +08:00
else:
if not group_manager.get_plugin_status(module, group_id):
return f"功能 {cmd} 正处于关闭状态!不要重复关闭."
group_manager.block_plugin(module, group_id)
if group_help_file.exists():
group_help_file.unlink()
if is_super:
for file in os.listdir(DATA_PATH / "group_help"):
file = DATA_PATH / "group_help" / file
file.unlink()
else:
_help_image = DATA_PATH / "group_help" / f"{group_id}.png"
if _help_image.exists():
_help_image.unlink()
return f"{status} {cmd} 功能!"
def set_plugin_status(cmd: str, block_type: str = "all"):
"""
说明:
设置插件功能状态超级用户使用
参数:
:param cmd: 功能名称
:param block_type: 限制类型, 'all': 私聊+群里, 'private': 私聊, 'group': 群聊
2022-02-19 18:20:19 +08:00
"""
status = cmd[:2]
cmd = cmd[2:]
module = plugins2settings_manager.get_plugin_module(cmd)
if status == "开启":
plugins_manager.unblock_plugin(module)
else:
plugins_manager.block_plugin(module, block_type=block_type)
for file in os.listdir(DATA_PATH / "group_help"):
file = DATA_PATH / "group_help" / file
file.unlink()
async def get_plugin_status():
"""
说明:
获取功能状态
2022-02-19 18:20:19 +08:00
"""
return await asyncio.get_event_loop().run_in_executor(None, _get_plugin_status)
def _get_plugin_status() -> MessageSegment:
"""
说明:
合成功能状态图片
2022-02-19 18:20:19 +08:00
"""
rst = "\t功能\n"
flag_str = "状态".rjust(4) + "\n"
2022-02-23 13:51:19 +08:00
tmp_name = []
for matcher in get_matchers():
if matcher.plugin_name not in tmp_name:
tmp_name.append(matcher.plugin_name)
module = matcher.plugin_name
flag = plugins_manager.get_plugin_block_type(module)
flag = flag.upper() + " CLOSE" if flag else "OPEN"
try:
2022-11-21 20:43:41 +08:00
plugin_name = plugins_manager.get(module).plugin_name
2022-02-23 13:51:19 +08:00
if (
"[Hidden]" in plugin_name
or "[Admin]" in plugin_name
or "[Superuser]" in plugin_name
2022-02-23 13:51:19 +08:00
):
continue
rst += f"{plugin_name}"
except KeyError:
rst += f"{module}"
2022-11-21 20:43:41 +08:00
if plugins_manager.get(module).error:
2022-02-23 13:51:19 +08:00
rst += "[ERROR]"
rst += "\n"
flag_str += f"{flag}\n"
2022-02-19 18:20:19 +08:00
height = len(rst.split("\n")) * 24
a = BuildImage(250, height, font_size=20)
a.text((10, 10), rst)
b = BuildImage(200, height, font_size=20)
b.text((10, 10), flag_str)
A = BuildImage(500, height)
A.paste(a)
A.paste(b, (270, 0))
return image(b64=A.pic2bs4())
async def update_member_info(group_id: int, remind_superuser: bool = False) -> bool:
"""
说明:
更新群成员信息
参数:
:param group_id: 群号
:param remind_superuser: 失败信息提醒超级用户
2022-02-19 18:20:19 +08:00
"""
bot = get_bot()
_group_user_list = await bot.get_group_member_list(group_id=group_id)
_error_member_list = []
_exist_member_list = []
# try:
for user_info in _group_user_list:
2022-05-21 22:40:05 +08:00
nickname = user_info["card"] or user_info["nickname"]
2022-02-19 18:20:19 +08:00
async with db.transaction():
# 更新权限
if user_info["role"] in [
2022-11-21 20:43:41 +08:00
"owner",
"admin",
] and not await LevelUser.is_group_flag(user_info["user_id"], group_id):
2022-02-19 18:20:19 +08:00
await LevelUser.set_level(
user_info["user_id"],
user_info["group_id"],
Config.get_config("admin_bot_manage", "ADMIN_DEFAULT_AUTH"),
)
if str(user_info["user_id"]) in bot.config.superusers:
await LevelUser.set_level(
user_info["user_id"], user_info["group_id"], 9
)
user = await GroupInfoUser.get_member_info(
user_info["user_id"], user_info["group_id"]
)
if user:
if user.user_name != nickname:
await user.update(user_name=nickname).apply()
logger.info(
f"用户{user_info['user_id']} 所属{user_info['group_id']} 更新群昵称成功"
)
_exist_member_list.append(int(user_info["user_id"]))
continue
join_time = datetime.strptime(
time.strftime(
"%Y-%m-%d %H:%M:%S", time.localtime(user_info["join_time"])
),
"%Y-%m-%d %H:%M:%S",
)
if await GroupInfoUser.add_member_info(
user_info["user_id"],
user_info["group_id"],
nickname,
join_time,
2022-02-19 18:20:19 +08:00
):
_exist_member_list.append(int(user_info["user_id"]))
logger.info(f"用户{user_info['user_id']} 所属{user_info['group_id']} 更新成功")
else:
_error_member_list.append(
f"用户{user_info['user_id']} 所属{user_info['group_id']} 更新失败\n"
)
_del_member_list = list(
set(_exist_member_list).difference(
set(await GroupInfoUser.get_group_member_id_list(group_id))
)
)
if _del_member_list:
for del_user in _del_member_list:
if await GroupInfoUser.delete_member_info(del_user, group_id):
logger.info(f"退群用户{del_user} 所属{group_id} 已删除")
else:
logger.info(f"退群用户{del_user} 所属{group_id} 删除失败")
if _error_member_list and remind_superuser:
result = ""
for error_user in _error_member_list:
result += error_user
await bot.send_private_msg(
user_id=int(list(bot.config.superusers)[0]), message=result[:-1]
)
return True
def set_group_bot_status(group_id: int, status: bool) -> str:
"""
说明:
设置群聊bot开关状态
参数:
:param group_id: 群号
:param status: 状态
2022-02-19 18:20:19 +08:00
"""
if status:
if group_manager.check_group_bot_status(group_id):
return "我还醒着呢!"
group_manager.turn_on_group_bot_status(group_id)
return "呜..醒来了..."
else:
group_manager.shutdown_group_bot_status(group_id)
return "那我先睡觉了..."