zhenxun_bot/zhenxun/builtin_plugins/__init__.py

173 lines
6.4 KiB
Python
Raw Normal View History

2024-10-03 23:56:17 +08:00
from datetime import datetime
import uuid
2024-02-25 03:18:34 +08:00
2024-10-03 23:56:17 +08:00
import nonebot
from nonebot.adapters import Bot
from nonebot.drivers import Driver
from tortoise import Tortoise
from tortoise.exceptions import IntegrityError, OperationalError
import ujson as json
2024-03-04 23:27:05 +08:00
from zhenxun.models.bot_connect_log import BotConnectLog
2024-10-21 19:07:35 +08:00
from zhenxun.models.bot_console import BotConsole
from zhenxun.models.goods_info import GoodsInfo
from zhenxun.models.group_member_info import GroupInfoUser
from zhenxun.models.sign_user import SignUser
2024-03-04 23:27:05 +08:00
from zhenxun.models.user_console import UserConsole
from zhenxun.services.log import logger
2024-03-04 23:27:05 +08:00
from zhenxun.utils.decorator.shop import shop_register
:sparkles: 首次启动时提供使用web ui方式完全配置 (#1870) * :sparkles: 添加全局优先级hook * :sparkles: 添加基础配置api * :sparkles: 添加数据库连接测试 * :speech_balloon: 提示重启 * :adhesive_bandage: 填充过配置时友好提示 * :bug: 首次生成简易配置后自动加载 * :sparkles: 添加配置后重启接口 * :sparkles: 添加重启标志文件 * :sparkles: 添加重启脚本命令 * :sparkles: 添加重启系统限制 * :sparkles: 首次配置判断是否为win系统 * :fire: 移除bat * :sparkles: 添加关于菜单 * :sparkles: 支持整合包插件安装和添加整合包文档 * :adhesive_bandage: 检测数据库路径 * :adhesive_bandage: 修改数据库路径检测 * :adhesive_bandage: 修改数据库路径检测 * :adhesive_bandage: 修复路径注入 * :art: 显示添加优先级 * :bug: 修改PriorityLifecycle字典类名称 * :zap: 修复路径问题 * :zap: 修复路径检测 * ✨ 新增路径验证功能,确保用户输入的路径安全并在项目根目录内 * ✨ 优化路径验证功能,增加对非法字符和路径长度的检查,确保用户输入的路径更加安全 * :rotating_light: auto fix by pre-commit hooks * ✨ 优化获取文件列表的代码格式 * :memo: 修改README中webui示例图 * ✨ 更新PriorityLifecycle.on_startup装饰器 * ✨ 简化安装依赖的命令构建逻辑 * :rotating_light: auto fix by pre-commit hooks --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
2025-06-16 09:11:41 +08:00
from zhenxun.utils.manager.priority_manager import PriorityLifecycle
from zhenxun.utils.manager.resource_manager import ResourceManager
from zhenxun.utils.platform import PlatformUtils
2024-02-04 04:18:54 +08:00
2024-03-05 08:29:46 +08:00
driver: Driver = nonebot.get_driver()
2024-03-04 23:27:05 +08:00
2024-10-03 23:56:17 +08:00
@driver.on_bot_connect
async def _(bot: Bot):
logger.debug(f"Bot: {bot.self_id} 建立连接...")
await BotConnectLog.create(
bot_id=bot.self_id, platform=bot.adapter, connect_time=datetime.now(), type=1
)
2024-10-21 19:07:35 +08:00
if not await BotConsole.exists(bot_id=bot.self_id):
try:
await BotConsole.create(
bot_id=bot.self_id, platform=PlatformUtils.get_platform(bot)
)
except IntegrityError as e:
logger.warning(f"记录bot: {bot.self_id} 数据已存在...", e=e)
2024-10-03 23:56:17 +08:00
@driver.on_bot_disconnect
async def _(bot: Bot):
logger.debug(f"Bot: {bot.self_id} 断开连接...")
try:
await BotConnectLog.create(
bot_id=bot.self_id,
platform=bot.adapter,
connect_time=datetime.now(),
type=0,
)
except Exception as e:
logger.warning(f"记录bot: {bot.self_id} 断开连接失败", e=e)
2024-10-03 23:56:17 +08:00
2024-03-04 23:27:05 +08:00
SIGN_SQL = """
SELECT user_id, checkin_count, add_probability, specify_probability, impression
FROM (
SELECT
t1.user_id,
t1.checkin_count,
t1.add_probability,
t1.specify_probability,
t1.impression,
ROW_NUMBER() OVER(PARTITION BY t1.user_id ORDER BY t1.impression DESC) AS rn
FROM sign_group_users t1
INNER JOIN (
SELECT user_id, MAX(impression) AS max_impression
FROM sign_group_users
GROUP BY user_id
) t2 ON t2.user_id = t1.user_id AND t2.max_impression = t1.impression
) t
WHERE rn = 1
2024-03-04 23:27:05 +08:00
"""
BAG_SQL = """
select t1.user_id, t1.gold, t1.property
from bag_users t1
2024-10-03 23:56:17 +08:00
join (
2024-03-04 23:27:05 +08:00
select user_id, max(t2.gold) as max_gold
from bag_users t2
2024-03-04 23:27:05 +08:00
group by user_id
) t on t.user_id = t1.user_id and t.max_gold = t1.gold
"""
:sparkles: 首次启动时提供使用web ui方式完全配置 (#1870) * :sparkles: 添加全局优先级hook * :sparkles: 添加基础配置api * :sparkles: 添加数据库连接测试 * :speech_balloon: 提示重启 * :adhesive_bandage: 填充过配置时友好提示 * :bug: 首次生成简易配置后自动加载 * :sparkles: 添加配置后重启接口 * :sparkles: 添加重启标志文件 * :sparkles: 添加重启脚本命令 * :sparkles: 添加重启系统限制 * :sparkles: 首次配置判断是否为win系统 * :fire: 移除bat * :sparkles: 添加关于菜单 * :sparkles: 支持整合包插件安装和添加整合包文档 * :adhesive_bandage: 检测数据库路径 * :adhesive_bandage: 修改数据库路径检测 * :adhesive_bandage: 修改数据库路径检测 * :adhesive_bandage: 修复路径注入 * :art: 显示添加优先级 * :bug: 修改PriorityLifecycle字典类名称 * :zap: 修复路径问题 * :zap: 修复路径检测 * ✨ 新增路径验证功能,确保用户输入的路径安全并在项目根目录内 * ✨ 优化路径验证功能,增加对非法字符和路径长度的检查,确保用户输入的路径更加安全 * :rotating_light: auto fix by pre-commit hooks * ✨ 优化获取文件列表的代码格式 * :memo: 修改README中webui示例图 * ✨ 更新PriorityLifecycle.on_startup装饰器 * ✨ 简化安装依赖的命令构建逻辑 * :rotating_light: auto fix by pre-commit hooks --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
2025-06-16 09:11:41 +08:00
@PriorityLifecycle.on_startup(priority=5)
2024-03-04 23:27:05 +08:00
async def _():
await ResourceManager.init_resources()
2024-07-30 22:36:09 +08:00
"""签到与用户的数据迁移"""
2024-07-21 23:26:56 +08:00
if goods_list := await GoodsInfo.filter(uuid__isnull=True).all():
for goods in goods_list:
goods.uuid = uuid.uuid1() # type: ignore
await GoodsInfo.bulk_update(goods_list, ["uuid"], 10)
2024-03-04 23:27:05 +08:00
await shop_register.load_register()
if (
2024-07-30 22:36:09 +08:00
not await UserConsole.annotate().count()
2024-03-04 23:27:05 +08:00
and not await SignUser.annotate().count()
):
2024-05-04 13:48:12 +08:00
try:
2024-08-15 20:27:19 +08:00
group_user = []
try:
group_user = await GroupInfoUser.filter(uid__isnull=False).all()
2024-08-24 12:30:49 +08:00
except Exception as e:
logger.warning("获取GroupInfoUser数据uid失败...", e=e)
2024-07-30 00:06:29 +08:00
user2uid = {u.user_id: u.uid for u in group_user}
2024-05-04 13:48:12 +08:00
db = Tortoise.get_connection("default")
old_sign_list = await db.execute_query_dict(SIGN_SQL)
old_bag_list = await db.execute_query_dict(BAG_SQL)
goods = {
g["goods_name"]: g["uuid"]
for g in await GoodsInfo.annotate().values("goods_name", "uuid")
}
create_list = []
sign_id_list = []
2024-10-03 23:56:17 +08:00
max_uid = max(user2uid.values()) + 1 if user2uid else 0
2024-05-04 13:48:12 +08:00
for old_sign in old_sign_list:
sign_id_list.append(old_sign["user_id"])
2024-10-03 23:56:17 +08:00
if old_bag := [
2024-05-04 13:48:12 +08:00
b for b in old_bag_list if b["user_id"] == old_sign["user_id"]
2024-10-03 23:56:17 +08:00
]:
2024-05-04 13:48:12 +08:00
old_bag = old_bag[0]
property = json.loads(old_bag["property"])
props = {}
if property:
for name, num in property.items():
if name in goods:
props[goods[name]] = num
create_list.append(
UserConsole(
user_id=old_sign["user_id"],
platform="qq",
2024-07-30 00:06:29 +08:00
uid=user2uid.get(old_sign["user_id"]) or max_uid,
2024-05-04 13:48:12 +08:00
props=props,
gold=old_bag["gold"],
)
)
2024-07-30 00:06:29 +08:00
if not user2uid.get(old_sign["user_id"]):
max_uid += 1
2024-05-04 13:48:12 +08:00
else:
create_list.append(
2024-07-30 00:06:29 +08:00
UserConsole(
user_id=old_sign["user_id"], platform="qq", uid=max_uid
)
2024-05-04 13:48:12 +08:00
)
2024-07-30 00:06:29 +08:00
max_uid += 1
2024-05-04 13:48:12 +08:00
if create_list:
logger.info("开始迁移用户数据...")
await UserConsole.bulk_create(create_list, 10)
logger.info("迁移用户数据完成!")
create_list.clear()
uc_dict = {u.user_id: u for u in await UserConsole.all()}
for old_sign in old_sign_list:
2024-10-03 23:56:17 +08:00
user_console = uc_dict.get(
old_sign["user_id"]
) or await UserConsole.get_user(old_sign["user_id"], "qq")
2024-03-04 23:27:05 +08:00
create_list.append(
2024-05-04 13:48:12 +08:00
SignUser(
2024-03-04 23:27:05 +08:00
user_id=old_sign["user_id"],
2024-05-04 13:48:12 +08:00
user_console=user_console,
2024-03-04 23:27:05 +08:00
platform="qq",
2024-05-04 13:48:12 +08:00
sign_count=old_sign["checkin_count"],
impression=old_sign["impression"],
add_probability=old_sign["add_probability"],
specify_probability=old_sign["specify_probability"],
2024-03-04 23:27:05 +08:00
)
)
2024-05-04 13:48:12 +08:00
if create_list:
logger.info("开始迁移签到数据...")
await SignUser.bulk_create(create_list, 10)
logger.info("迁移签到数据完成!")
except OperationalError as e:
logger.warning("数据迁移", e=e)