mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 06:12:53 +08:00
209 lines
7.3 KiB
Python
Executable File
209 lines
7.3 KiB
Python
Executable File
from asyncpg.exceptions import (
|
||
DuplicateColumnError,
|
||
UndefinedColumnError,
|
||
PostgresSyntaxError,
|
||
)
|
||
from nonebot import Driver
|
||
from services.db_context import db
|
||
from models.group_info import GroupInfo
|
||
from models.bag_user import BagUser
|
||
from nonebot.adapters.cqhttp import Bot
|
||
from services.log import logger
|
||
from configs.path_config import TEXT_PATH
|
||
from asyncio.exceptions import TimeoutError
|
||
from typing import List
|
||
from utils.http_utils import AsyncHttpx
|
||
from utils.utils import scheduler
|
||
from pathlib import Path
|
||
import nonebot
|
||
|
||
try:
|
||
import ujson as json
|
||
except ModuleNotFoundError:
|
||
import json
|
||
|
||
|
||
driver: Driver = nonebot.get_driver()
|
||
|
||
|
||
@driver.on_startup
|
||
async def update_city():
|
||
"""
|
||
部分插件需要中国省份城市
|
||
这里直接更新,避免插件内代码重复
|
||
"""
|
||
china_city = Path(TEXT_PATH) / "china_city.json"
|
||
data = {}
|
||
if not china_city.exists():
|
||
try:
|
||
res = await AsyncHttpx.get(
|
||
"http://www.weather.com.cn/data/city3jdata/china.html", timeout=5
|
||
)
|
||
res.encoding = "utf8"
|
||
provinces_data = json.loads(res.text)
|
||
for province in provinces_data.keys():
|
||
data[provinces_data[province]] = []
|
||
res = await AsyncHttpx.get(
|
||
f"http://www.weather.com.cn/data/city3jdata/provshi/{province}.html",
|
||
timeout=5,
|
||
)
|
||
res.encoding = "utf8"
|
||
city_data = json.loads(res.text)
|
||
for city in city_data.keys():
|
||
data[provinces_data[province]].append(city_data[city])
|
||
with open(china_city, "w", encoding="utf8") as f:
|
||
json.dump(data, f, indent=4, ensure_ascii=False)
|
||
logger.info("自动更新城市列表完成.....")
|
||
except TimeoutError:
|
||
logger.warning("自动更新城市列表超时.....")
|
||
except ValueError:
|
||
logger.warning("自动城市列表失败.....")
|
||
except Exception as e:
|
||
logger.error(f"自动城市列表未知错误 {type(e)}:{e}")
|
||
|
||
|
||
@driver.on_startup
|
||
async def _():
|
||
"""
|
||
数据库表结构变换
|
||
"""
|
||
_flag = []
|
||
sql_str = [
|
||
(
|
||
"ALTER TABLE group_info ADD group_flag Integer NOT NULL DEFAULT 0;",
|
||
"group_info",
|
||
), # group_info表添加一个group_flag
|
||
(
|
||
"ALTER TABLE bag_users rename belonging_group To group_id;",
|
||
"bag_users",
|
||
), # 将 bag_users 的 belonging_group 改为 group_id
|
||
(
|
||
"ALTER TABLE group_info_users rename belonging_group To group_id;",
|
||
"group_info_users",
|
||
),
|
||
(
|
||
"ALTER TABLE sign_group_users rename belonging_group To group_id;",
|
||
"sign_group_users",
|
||
),
|
||
(
|
||
"ALTER TABLE open_cases_users rename belonging_group To group_id;",
|
||
"open_cases_users",
|
||
),
|
||
(
|
||
"ALTER TABLE bag_users ADD property json NOT NULL DEFAULT '{}';",
|
||
"bag_users",
|
||
), # bag_users 新增字段 property 替代 props
|
||
(
|
||
"ALTER TABLE genshin ADD auto_sign_time timestamp with time zone;",
|
||
"genshin"
|
||
), # 新增原神自动签到字段
|
||
(
|
||
"ALTER TABLE genshin ADD resin_remind boolean DEFAULT False;",
|
||
"genshin"
|
||
), # 新增原神自动签到字段
|
||
(
|
||
"ALTER TABLE genshin ADD resin_recovery_time timestamp with time zone;",
|
||
"genshin"
|
||
), # 新增原神自动签到字段
|
||
]
|
||
for sql in sql_str:
|
||
try:
|
||
flag = sql[1]
|
||
sql = sql[0]
|
||
query = db.text(sql)
|
||
await db.first(query)
|
||
logger.info(f"完成sql操作:{sql}")
|
||
_flag.append(flag)
|
||
except (DuplicateColumnError, UndefinedColumnError):
|
||
pass
|
||
except PostgresSyntaxError:
|
||
logger.error(f"语法错误:执行sql失败:{sql}")
|
||
# bag_user 将文本转为字典格式
|
||
await __database_script(_flag)
|
||
|
||
# 完成后
|
||
end_sql_str = [
|
||
# "ALTER TABLE bag_users DROP COLUMN props;" # 删除 bag_users 的 props 字段(还不到时候)
|
||
]
|
||
for sql in end_sql_str:
|
||
try:
|
||
query = db.text(sql)
|
||
await db.first(query)
|
||
logger.info(f"完成执行sql操作:{sql}")
|
||
except (DuplicateColumnError, UndefinedColumnError):
|
||
pass
|
||
except PostgresSyntaxError:
|
||
logger.error(f"语法错误:执行sql失败:{sql}")
|
||
|
||
# str2json_sql = ["alter table bag_users alter COLUMN props type json USING props::json;"] # 字段类型替换
|
||
# rename_sql = 'alter table {} rename {} to {};' # 字段更名
|
||
# for sql in str2json_sql:
|
||
# try:
|
||
# query = db.text(sql)
|
||
# await db.first(query)
|
||
# except DuplicateColumnError:
|
||
# pass
|
||
|
||
|
||
@driver.on_bot_connect
|
||
async def _(bot: Bot):
|
||
"""
|
||
版本某些需要的变换
|
||
"""
|
||
# 清空不存在的群聊信息,并将已所有已存在的群聊group_flag设置为1(认证所有已存在的群)
|
||
if not await GroupInfo.get_group_info(114514):
|
||
# 标识符,该功能只需执行一次
|
||
await GroupInfo.add_group_info(114514, "114514", 114514, 114514, 1)
|
||
group_list = await bot.get_group_list()
|
||
group_list = [g["group_id"] for g in group_list]
|
||
_gl = [x.group_id for x in await GroupInfo.get_all_group()]
|
||
if 114514 in _gl:
|
||
_gl.remove(114514)
|
||
for group_id in _gl:
|
||
if group_id in group_list:
|
||
if await GroupInfo.get_group_info(group_id):
|
||
await GroupInfo.set_group_flag(group_id, 1)
|
||
else:
|
||
group_info = await bot.get_group_info(group_id=group_id)
|
||
await GroupInfo.add_group_info(
|
||
group_info["group_id"],
|
||
group_info["group_name"],
|
||
group_info["max_member_count"],
|
||
group_info["member_count"],
|
||
1,
|
||
)
|
||
logger.info(f"已将群聊 {group_id} 添加认证...")
|
||
else:
|
||
await GroupInfo.delete_group_info(group_id)
|
||
logger.info(f"移除不存在的群聊信息:{group_id}")
|
||
|
||
|
||
async def __database_script(_flag: List[str]):
|
||
# bag_user 将文本转为字典格式
|
||
if "bag_users" in _flag:
|
||
for x in await BagUser.get_all_users():
|
||
props = {}
|
||
if x.props:
|
||
for prop in [p for p in x.props.split(",") if p]:
|
||
if props.get(prop):
|
||
props[prop] += 1
|
||
else:
|
||
props[prop] = 1
|
||
logger.info(
|
||
f"__database_script USER {x.user_qq} GROUP {x.group_id} 更新数据 {props}"
|
||
)
|
||
await x.update(
|
||
property=props,
|
||
props="",
|
||
).apply()
|
||
|
||
|
||
# 自动更新城市列表
|
||
@scheduler.scheduled_job(
|
||
"cron",
|
||
hour=6,
|
||
minute=1,
|
||
)
|
||
async def _():
|
||
await update_city()
|