mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 14:22:55 +08:00
modified: basic_plugins/admin_bot_manage/admin_config.py modified: basic_plugins/admin_bot_manage/custom_welcome_message.py modified: basic_plugins/admin_bot_manage/timing_task.py modified: basic_plugins/apscheduler/__init__.py modified: basic_plugins/ban/__init__.py modified: basic_plugins/ban/data_source.py modified: basic_plugins/chat_history/chat_message.py modified: basic_plugins/chat_history/chat_message_handle.py modified: basic_plugins/group_handle/__init__.py modified: basic_plugins/hooks/_utils.py modified: basic_plugins/hooks/ban_hook.py modified: basic_plugins/hooks/chkdsk_hook.py modified: basic_plugins/init_plugin_config/__init__.py deleted: basic_plugins/init_plugin_config/init_group_manager.py modified: basic_plugins/invite_manager/__init__.py new file: basic_plugins/invite_manager/utils.py modified: basic_plugins/nickname.py modified: basic_plugins/plugin_shop/__init__.py modified: basic_plugins/plugin_shop/data_source.py modified: basic_plugins/scripts.py modified: basic_plugins/shop/__init__.py modified: basic_plugins/shop/buy.py modified: basic_plugins/shop/gold.py modified: basic_plugins/shop/my_props/__init__.py deleted: basic_plugins/shop/reset_today_gold.py modified: basic_plugins/shop/shop_handle/__init__.py modified: basic_plugins/shop/shop_handle/data_source.py modified: basic_plugins/shop/use/__init__.py modified: basic_plugins/shop/use/data_source.py modified: basic_plugins/super_cmd/__init__.py modified: basic_plugins/super_cmd/bot_friend_group.py modified: basic_plugins/super_cmd/clear_data.py modified: basic_plugins/super_cmd/exec_sql.py modified: basic_plugins/super_cmd/manager_group.py modified: basic_plugins/super_cmd/reload_setting.py modified: basic_plugins/super_cmd/set_admin_permissions.py deleted: basic_plugins/super_cmd/super_task_switch.py modified: basic_plugins/super_cmd/update_friend_group_info.py modified: basic_plugins/super_help/__init__.py modified: basic_plugins/update_info.py modified: configs/config.py modified: configs/utils/__init__.py modified: models/bag_user.py modified: models/ban_user.py modified: models/chat_history.py modified: models/friend_user.py modified: models/goods_info.py modified: models/group_info.py modified: models/group_member_info.py modified: models/level_user.py modified: models/sign_group_user.py modified: models/user_shop_gold_log.py modified: plugins/aconfig/__init__.py modified: plugins/ai/__init__.py modified: plugins/ai/data_source.py modified: plugins/bilibili_sub/__init__.py modified: plugins/bilibili_sub/data_source.py modified: plugins/bilibili_sub/model.py modified: plugins/black_word/__init__.py modified: plugins/black_word/model.py modified: plugins/black_word/utils.py modified: plugins/bt/data_source.py modified: plugins/genshin/almanac/__init__.py modified: plugins/genshin/material_remind/__init__.py modified: plugins/genshin/query_user/_models/__init__.py modified: plugins/genshin/query_user/_utils/__init__.py modified: plugins/genshin/query_user/bind/__init__.py modified: plugins/genshin/query_user/genshin_sign/__init__.py modified: plugins/genshin/query_user/genshin_sign/data_source.py modified: plugins/genshin/query_user/genshin_sign/init_task.py modified: plugins/genshin/query_user/mihoyobbs_sign/__init__.py modified: plugins/genshin/query_user/query_memo/__init__.py modified: plugins/genshin/query_user/query_memo/data_source.py modified: plugins/genshin/query_user/query_role/__init__.py modified: plugins/genshin/query_user/query_role/data_source.py modified: plugins/genshin/query_user/reset_today_query_user_data/__init__.py modified: plugins/genshin/query_user/resin_remind/__init__.py modified: plugins/genshin/query_user/resin_remind/init_task.py modified: plugins/gold_redbag/model.py modified: plugins/image_management/send_image/__init__.py modified: plugins/my_info/__init__.py modified: plugins/open_cases/models/buff_prices.py modified: plugins/open_cases/models/open_cases_user.py modified: plugins/open_cases/open_cases_c.py modified: plugins/open_cases/utils.py modified: plugins/parse_bilibili_json.py modified: plugins/pid_search.py modified: plugins/pix_gallery/__init__.py modified: plugins/pix_gallery/_data_source.py modified: plugins/pix_gallery/_model/omega_pixiv_illusts.py modified: plugins/pix_gallery/_model/pixiv.py modified: plugins/pix_gallery/_model/pixiv_keyword_user.py modified: plugins/pix_gallery/pix_add_keyword.py modified: plugins/pix_gallery/pix_pass_del_keyword.py modified: plugins/pix_gallery/pix_show_info.py modified: plugins/pix_gallery/pix_update.py modified: plugins/pixiv_rank_search/data_source.py modified: plugins/poke/__init__.py modified: plugins/russian/__init__.py modified: plugins/russian/data_source.py modified: plugins/russian/model.py modified: plugins/send_dinggong_voice/__init__.py modified: plugins/send_setu_/_model.py modified: plugins/send_setu_/send_setu/__init__.py modified: plugins/send_setu_/send_setu/data_source.py modified: plugins/send_setu_/update_setu/data_source.py modified: plugins/sign_in/goods_register.py modified: plugins/sign_in/group_user_checkin.py modified: plugins/sign_in/random_event.py modified: plugins/sign_in/utils.py modified: plugins/statistics/_model.py modified: plugins/statistics/statistics_handle.py modified: plugins/statistics/statistics_hook.py modified: plugins/update_picture.py modified: plugins/web_ui/api/request.py modified: plugins/word_bank/_model.py deleted: plugins/word_bank/_old_model.py modified: plugins/word_bank/_rule.py modified: plugins/word_bank/word_handle.py modified: plugins/word_clouds/data_source.py modified: resources/image/sign/sign_res/bar.png modified: resources/image/sign/sign_res/bar_white.png modified: services/db_context.py modified: services/log.py modified: utils/browser.py modified: utils/data_utils.py modified: utils/depends/__init__.py modified: utils/http_utils.py modified: utils/image_utils.py modified: utils/manager/admin_manager.py modified: utils/message_builder.py modified: utils/utils.py
202 lines
7.6 KiB
Python
202 lines
7.6 KiB
Python
import os
|
||
import shutil
|
||
import zipfile
|
||
from pathlib import Path
|
||
from typing import Tuple, Union
|
||
|
||
import ujson as json
|
||
|
||
from configs.path_config import DATA_PATH, TEMP_PATH
|
||
from services import logger
|
||
from utils.http_utils import AsyncHttpx
|
||
from utils.image_utils import BuildImage, text2image
|
||
from utils.manager import plugins_manager
|
||
from utils.utils import is_number
|
||
|
||
path = DATA_PATH / "plugin_shop"
|
||
if not path.exists():
|
||
path.mkdir(parents=True, exist_ok=True)
|
||
|
||
repo_json_url = "https://github.com/zhenxun-org/nonebot_plugins_zhenxun_bot/archive/refs/heads/index.zip"
|
||
repo_zip_path = TEMP_PATH / "plugin_repo_json.zip"
|
||
plugin_json = path / "zhenxun_plugins.json"
|
||
|
||
extensive_plugin_path = Path() / "extensive_plugin"
|
||
|
||
path = DATA_PATH / "plugin_shop"
|
||
if not path.exists():
|
||
path.mkdir(parents=True, exist_ok=True)
|
||
if not extensive_plugin_path.exists():
|
||
extensive_plugin_path.mkdir(parents=True, exist_ok=True)
|
||
|
||
data = {}
|
||
|
||
|
||
async def install_plugin(name: str) -> str:
|
||
"""
|
||
安装插件
|
||
:param name: 插件名或下标
|
||
"""
|
||
try:
|
||
if is_number(name):
|
||
name, code = await get_plugin_name(int(name))
|
||
if code != 200:
|
||
return name
|
||
plugin_url = data[name]["download_url"]
|
||
url = (await AsyncHttpx.get(plugin_url)).headers.get("Location")
|
||
zip_file = TEMP_PATH / f"{name}.zip"
|
||
if zip_file.exists():
|
||
zip_file.unlink()
|
||
if await AsyncHttpx.download_file(url, zip_file):
|
||
logger.debug("开始解压插件压缩包...", "安装插件", target=name)
|
||
# 解压
|
||
zf = zipfile.ZipFile(zip_file, "r")
|
||
extract_path = TEMP_PATH / f"{name}"
|
||
if extract_path.exists():
|
||
shutil.rmtree(extract_path.absolute(), ignore_errors=True)
|
||
extract_path.mkdir(exist_ok=True, parents=True)
|
||
for file in zf.namelist():
|
||
zf.extract(file, extract_path)
|
||
zf.close()
|
||
logger.debug("解压插件压缩包完成...", "安装插件", target=name)
|
||
logger.debug("开始移动插件文件夹...", "安装插件", target=name)
|
||
if (extensive_plugin_path / f"{name}").exists():
|
||
logger.debug(
|
||
"extensive_plugin目录下文件夹已存在,删除该目录插件文件夹...", "安装插件", target=name
|
||
)
|
||
shutil.rmtree(
|
||
(extensive_plugin_path / f"{name}").absolute(), ignore_errors=True
|
||
)
|
||
extract_path.rename(extensive_plugin_path / f"{name}")
|
||
prompt = ""
|
||
if "pyproject.toml" in os.listdir(extensive_plugin_path / f"{name}"):
|
||
prompt = "检测到该插件含有额外依赖,当前安装无法保证依赖完全安装成功。"
|
||
os.system(
|
||
f"poetry run pip install -r {(extensive_plugin_path / f'{name}' / 'pyproject.toml').absolute()}"
|
||
)
|
||
elif "requirements.txt" in os.listdir(extensive_plugin_path / f"{name}"):
|
||
prompt = "检测到该插件含有额外依赖,当前安装无法保证依赖完全安装成功。"
|
||
os.system(
|
||
f"poetry run pip install -r {(extensive_plugin_path / f'{name}' / 'requirements.txt').absolute()}"
|
||
)
|
||
with open(extensive_plugin_path / f"{name}" / "plugin_info.json", "w") as f:
|
||
json.dump(data[name], f, ensure_ascii=False, indent=4)
|
||
logger.debug("移动插件文件夹完成...", "安装插件", target=name)
|
||
logger.info(f"成功安装插件 {name} 成功!\n{prompt}", "安装插件", target=name)
|
||
return f"成功安装插件 {name},请重启真寻!"
|
||
except Exception as e:
|
||
logger.error(f"安装插失败", "安装插件", target=name, e=e)
|
||
return f"安装插件 {name} 失败 {type(e)}:{e}"
|
||
|
||
|
||
async def uninstall_plugin(name: str) -> str:
|
||
"""
|
||
删除插件
|
||
:param name: 插件名或下标
|
||
"""
|
||
try:
|
||
if is_number(name):
|
||
name, code = await get_plugin_name(int(name))
|
||
if code != 200:
|
||
return name
|
||
if name not in os.listdir(extensive_plugin_path):
|
||
return f"未安装 {name} 插件!"
|
||
shutil.rmtree((extensive_plugin_path / name).absolute(), ignore_errors=True)
|
||
logger.info(f"插件 {name} 删除成功!")
|
||
return f"插件 {name} 删除成功!"
|
||
except Exception as e:
|
||
logger.error(f"删除插件 {name} 失败 {type(e)}:{e}")
|
||
return f"删除插件 {name} 失败 {type(e)}:{e}"
|
||
|
||
|
||
async def show_plugin_repo() -> Union[int, str]:
|
||
"""
|
||
获取插件仓库数据并格式化
|
||
"""
|
||
if not plugin_json.exists():
|
||
code = await download_json()
|
||
if code != 200:
|
||
return code
|
||
plugin_info = json.load(open(plugin_json, "r", encoding="utf8"))
|
||
plugins_data = plugins_manager.get_data()
|
||
load_plugin_list = plugins_data.keys()
|
||
image_list = []
|
||
w, h = 0, 0
|
||
line_height = 10
|
||
for i, key in enumerate(plugin_info.keys()):
|
||
data[key] = {
|
||
"名称": plugin_info[key]["plugin_name"],
|
||
"模块": key,
|
||
"作者": plugin_info[key]["author"],
|
||
"版本": plugin_info[key]["version"],
|
||
"简介": plugin_info[key]["introduction"],
|
||
"download_url": plugin_info[key]["download_url"],
|
||
"github_url": plugin_info[key]["github_url"],
|
||
}
|
||
status = ""
|
||
version = ""
|
||
if key in load_plugin_list:
|
||
status = "<f font_color=#1a7e30>[已安装]</f>"
|
||
version = f"<f font_color=#1a7e30>[{plugins_data[key].version}]</f>"
|
||
s = (
|
||
f'id:{i+1}\n名称:{plugin_info[key]["plugin_name"]}'
|
||
f" \t\t{status}\n"
|
||
f"模块:{key}\n"
|
||
f'作者:{plugin_info[key]["author"]}\n'
|
||
f'版本:{plugin_info[key]["version"]} \t\t{version}\n'
|
||
f'简介:{plugin_info[key]["introduction"]}\n'
|
||
f"-------------------"
|
||
)
|
||
img = await text2image(s, font_size=20, color="#f9f6f2")
|
||
w = w if w > img.w else img.w
|
||
h += img.h + line_height
|
||
image_list.append(img)
|
||
A = BuildImage(w + 50, h + 50, color="#f9f6f2")
|
||
cur_h = 25
|
||
for img in image_list:
|
||
await A.apaste(img, (25, cur_h))
|
||
cur_h += img.h + line_height
|
||
return A.pic2bs4()
|
||
|
||
|
||
async def download_json() -> int:
|
||
"""
|
||
下载插件库json文件
|
||
"""
|
||
try:
|
||
url = (await AsyncHttpx.get(repo_json_url)).headers.get("Location")
|
||
if repo_zip_path.exists():
|
||
repo_zip_path.unlink()
|
||
if await AsyncHttpx.download_file(url, repo_zip_path):
|
||
zf = zipfile.ZipFile(repo_zip_path, "r")
|
||
extract_path = path / "temp"
|
||
for file in zf.namelist():
|
||
zf.extract(file, extract_path)
|
||
zf.close()
|
||
if plugin_json.exists():
|
||
plugin_json.unlink()
|
||
(
|
||
extract_path
|
||
/ "nonebot_plugins_zhenxun_bot-index"
|
||
/ "zhenxun_plugins.json"
|
||
).rename(plugin_json)
|
||
shutil.rmtree(extract_path.absolute(), ignore_errors=True)
|
||
return 200
|
||
except Exception as e:
|
||
logger.error(f"下载插件库压缩包失败或解压失败 {type(e)}:{e}")
|
||
return 999
|
||
|
||
|
||
async def get_plugin_name(name: int) -> Tuple[str, int]:
|
||
"""
|
||
通过下标获取插件名
|
||
:param name: 下标
|
||
"""
|
||
name = int(name)
|
||
if not data:
|
||
await show_plugin_repo()
|
||
if name < 1 or name > len(data.keys()):
|
||
return "下标超过上下限!", 999
|
||
name = list(data.keys())[name - 1]
|
||
return name, 200
|