2024-08-20 21:29:42 +08:00
|
|
|
import shutil
|
2024-08-21 16:37:07 +08:00
|
|
|
import subprocess
|
2024-08-24 19:32:52 +08:00
|
|
|
from pathlib import Path
|
2024-08-20 21:29:42 +08:00
|
|
|
|
|
|
|
|
import ujson as json
|
2024-09-03 13:04:49 +08:00
|
|
|
from aiocache import cached
|
2024-08-20 21:29:42 +08:00
|
|
|
|
2024-08-24 19:42:30 +08:00
|
|
|
from zhenxun.services.log import logger
|
|
|
|
|
from zhenxun.utils.http_utils import AsyncHttpx
|
2024-08-29 22:01:34 +08:00
|
|
|
from zhenxun.models.plugin_info import PluginInfo
|
|
|
|
|
from zhenxun.utils.image_utils import RowStyle, BuildImage, ImageTemplate
|
2024-09-02 01:50:39 +08:00
|
|
|
from zhenxun.builtin_plugins.auto_update.config import REQ_TXT_FILE_STRING
|
|
|
|
|
from zhenxun.builtin_plugins.plugin_store.models import (
|
|
|
|
|
FileInfo,
|
|
|
|
|
FileType,
|
|
|
|
|
RepoInfo,
|
|
|
|
|
JsdPackageInfo,
|
|
|
|
|
StorePluginInfo,
|
|
|
|
|
)
|
2024-08-20 21:29:42 +08:00
|
|
|
|
2024-08-26 19:43:12 +08:00
|
|
|
from .config import (
|
|
|
|
|
BASE_PATH,
|
2024-09-03 13:04:49 +08:00
|
|
|
EXTRA_GITHUB_URL,
|
2024-09-02 01:50:39 +08:00
|
|
|
DEFAULT_GITHUB_URL,
|
|
|
|
|
JSD_PACKAGE_API_FORMAT,
|
2024-08-26 19:43:12 +08:00
|
|
|
)
|
2024-08-20 21:29:42 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def row_style(column: str, text: str) -> RowStyle:
|
|
|
|
|
"""被动技能文本风格
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
column: 表头
|
|
|
|
|
text: 文本内容
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
RowStyle: RowStyle
|
|
|
|
|
"""
|
|
|
|
|
style = RowStyle()
|
2024-08-29 22:01:34 +08:00
|
|
|
if column == "-" and text == "已安装":
|
|
|
|
|
style.font_color = "#67C23A"
|
2024-08-20 21:29:42 +08:00
|
|
|
return style
|
|
|
|
|
|
|
|
|
|
|
2024-09-02 01:50:39 +08:00
|
|
|
def full_files_path(
|
|
|
|
|
jsd_package_info: JsdPackageInfo, module_path: str, is_dir: bool = True
|
|
|
|
|
) -> list[FileInfo]:
|
|
|
|
|
"""
|
|
|
|
|
获取文件路径
|
2024-08-20 21:29:42 +08:00
|
|
|
|
|
|
|
|
参数:
|
2024-09-02 01:50:39 +08:00
|
|
|
jsd_package_info: JsdPackageInfo
|
|
|
|
|
module_path: 模块路径
|
|
|
|
|
is_dir: 是否为目录
|
2024-08-20 21:29:42 +08:00
|
|
|
|
2024-09-02 01:50:39 +08:00
|
|
|
返回:
|
|
|
|
|
list[FileInfo]: 文件路径
|
2024-08-20 21:29:42 +08:00
|
|
|
"""
|
2024-09-02 01:50:39 +08:00
|
|
|
paths: list[str] = module_path.split(".")
|
|
|
|
|
cur_files: list[FileInfo] = jsd_package_info.files
|
|
|
|
|
for path in paths:
|
|
|
|
|
for cur_file in cur_files:
|
|
|
|
|
if (
|
|
|
|
|
cur_file.type == FileType.DIR
|
|
|
|
|
and cur_file.name == path
|
|
|
|
|
and cur_file.files
|
|
|
|
|
and (is_dir or path != paths[-1])
|
|
|
|
|
):
|
|
|
|
|
cur_files = cur_file.files
|
|
|
|
|
break
|
|
|
|
|
if not is_dir and path == paths[-1] and cur_file.name.split(".")[0] == path:
|
|
|
|
|
return cur_files
|
|
|
|
|
else:
|
|
|
|
|
raise ValueError(f"模块路径 {module_path} 不存在")
|
|
|
|
|
return cur_files
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def recurrence_files(
|
|
|
|
|
files: list[FileInfo], dir_path: str, is_dir: bool = True
|
|
|
|
|
) -> list[str]:
|
|
|
|
|
"""
|
|
|
|
|
递归获取文件路径
|
2024-08-20 21:29:42 +08:00
|
|
|
|
|
|
|
|
参数:
|
2024-09-02 01:50:39 +08:00
|
|
|
files: 文件列表
|
|
|
|
|
dir_path: 目录路径
|
|
|
|
|
is_dir: 是否为目录
|
2024-08-20 21:29:42 +08:00
|
|
|
|
2024-09-02 01:50:39 +08:00
|
|
|
返回:
|
|
|
|
|
list[str]: 文件路径
|
2024-08-20 21:29:42 +08:00
|
|
|
"""
|
2024-09-02 01:50:39 +08:00
|
|
|
paths = []
|
|
|
|
|
for file in files:
|
|
|
|
|
if is_dir and file.type == FileType.DIR and file.files:
|
|
|
|
|
paths.extend(
|
|
|
|
|
recurrence_files(file.files, f"{dir_path}/{file.name}", is_dir)
|
|
|
|
|
)
|
|
|
|
|
elif file.type == FileType.FILE:
|
|
|
|
|
if dir_path.endswith(file.name):
|
|
|
|
|
paths.append(dir_path)
|
|
|
|
|
elif is_dir:
|
|
|
|
|
paths.append(f"{dir_path}/{file.name}")
|
|
|
|
|
return paths
|
2024-08-20 21:29:42 +08:00
|
|
|
|
|
|
|
|
|
2024-08-21 16:37:07 +08:00
|
|
|
def install_requirement(plugin_path: Path):
|
2024-08-26 10:39:33 +08:00
|
|
|
requirement_files = ["requirement.txt", "requirements.txt"]
|
|
|
|
|
requirement_paths = [plugin_path / file for file in requirement_files]
|
2024-08-24 14:49:55 +08:00
|
|
|
|
2024-08-26 19:43:12 +08:00
|
|
|
existing_requirements = next(
|
|
|
|
|
(path for path in requirement_paths if path.exists()), None
|
|
|
|
|
)
|
2024-08-26 10:39:33 +08:00
|
|
|
|
|
|
|
|
if not existing_requirements:
|
2024-08-26 19:43:12 +08:00
|
|
|
logger.debug(
|
|
|
|
|
f"No requirement.txt found for plugin: {plugin_path.name}", "插件管理"
|
|
|
|
|
)
|
2024-08-21 16:37:07 +08:00
|
|
|
return
|
2024-08-24 14:49:55 +08:00
|
|
|
|
2024-08-21 16:37:07 +08:00
|
|
|
try:
|
2024-08-24 19:32:52 +08:00
|
|
|
result = subprocess.run(
|
2024-08-26 10:39:33 +08:00
|
|
|
["pip", "install", "-r", str(existing_requirements)],
|
2024-08-24 19:32:52 +08:00
|
|
|
check=True,
|
2024-08-29 22:01:34 +08:00
|
|
|
capture_output=True,
|
2024-08-24 19:32:52 +08:00
|
|
|
text=True,
|
|
|
|
|
)
|
|
|
|
|
logger.debug(
|
2024-08-29 22:01:34 +08:00
|
|
|
"Successfully installed dependencies for"
|
|
|
|
|
f" plugin: {plugin_path.name}. Output:\n{result.stdout}",
|
2024-08-24 19:32:52 +08:00
|
|
|
"插件管理",
|
|
|
|
|
)
|
2024-08-29 22:01:34 +08:00
|
|
|
except subprocess.CalledProcessError:
|
2024-08-24 19:32:52 +08:00
|
|
|
logger.error(
|
2024-08-29 22:01:34 +08:00
|
|
|
f"Failed to install dependencies for plugin: {plugin_path.name}. "
|
|
|
|
|
" Error:\n{e.stderr}"
|
2024-08-24 19:32:52 +08:00
|
|
|
)
|
2024-08-21 16:37:07 +08:00
|
|
|
|
|
|
|
|
|
2024-08-20 21:29:42 +08:00
|
|
|
class ShopManage:
|
|
|
|
|
@classmethod
|
2024-09-03 13:04:49 +08:00
|
|
|
@cached(60)
|
2024-09-02 01:50:39 +08:00
|
|
|
async def __get_data(cls) -> dict[str, StorePluginInfo]:
|
2024-08-20 21:29:42 +08:00
|
|
|
"""获取插件信息数据
|
|
|
|
|
|
|
|
|
|
异常:
|
|
|
|
|
ValueError: 访问请求失败
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
dict: 插件信息数据
|
|
|
|
|
"""
|
2024-09-03 13:04:49 +08:00
|
|
|
default_github_url = await RepoInfo.parse_github_url(
|
|
|
|
|
DEFAULT_GITHUB_URL
|
|
|
|
|
).get_download_url_with_path("plugins.json")
|
|
|
|
|
extra_github_url = await RepoInfo.parse_github_url(
|
|
|
|
|
EXTRA_GITHUB_URL
|
|
|
|
|
).get_download_url_with_path("plugins.json")
|
|
|
|
|
res = await AsyncHttpx.get(default_github_url)
|
|
|
|
|
res2 = await AsyncHttpx.get(extra_github_url)
|
2024-08-26 10:39:33 +08:00
|
|
|
|
|
|
|
|
# 检查请求结果
|
|
|
|
|
if res.status_code != 200 or res2.status_code != 200:
|
|
|
|
|
raise ValueError(f"下载错误, code: {res.status_code}, {res2.status_code}")
|
|
|
|
|
|
|
|
|
|
# 解析并合并返回的 JSON 数据
|
|
|
|
|
data1 = json.loads(res.text)
|
|
|
|
|
data2 = json.loads(res2.text)
|
2024-09-02 01:50:39 +08:00
|
|
|
return {
|
|
|
|
|
name: StorePluginInfo(**detail)
|
|
|
|
|
for name, detail in {**data1, **data2}.items()
|
|
|
|
|
}
|
2024-08-20 21:29:42 +08:00
|
|
|
|
2024-08-24 20:24:05 +08:00
|
|
|
@classmethod
|
2024-09-02 01:50:39 +08:00
|
|
|
def version_check(cls, plugin_info: StorePluginInfo, suc_plugin: dict[str, str]):
|
|
|
|
|
"""版本检查
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
plugin_info: StorePluginInfo
|
|
|
|
|
suc_plugin: dict[str, str]
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
str: 版本号
|
|
|
|
|
"""
|
|
|
|
|
module = plugin_info.module
|
2024-09-02 15:00:37 +08:00
|
|
|
if suc_plugin.get(module) and not cls.check_version_is_new(
|
|
|
|
|
plugin_info, suc_plugin
|
|
|
|
|
):
|
2024-09-02 01:50:39 +08:00
|
|
|
return f"{suc_plugin[module]} (有更新->{plugin_info.version})"
|
|
|
|
|
return plugin_info.version
|
2024-08-24 20:24:05 +08:00
|
|
|
|
|
|
|
|
@classmethod
|
2024-09-02 01:50:39 +08:00
|
|
|
def check_version_is_new(
|
|
|
|
|
cls, plugin_info: StorePluginInfo, suc_plugin: dict[str, str]
|
|
|
|
|
):
|
|
|
|
|
"""检查版本是否有更新
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
plugin_info: StorePluginInfo
|
|
|
|
|
suc_plugin: dict[str, str]
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
bool: 是否有更新
|
|
|
|
|
"""
|
|
|
|
|
module = plugin_info.module
|
2024-09-02 12:00:27 +08:00
|
|
|
return suc_plugin.get(module) and plugin_info.version == suc_plugin[module]
|
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
async def get_loaded_plugins(cls, *args) -> list[tuple[str, str]]:
|
|
|
|
|
"""获取已加载的插件
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
list[str]: 已加载的插件
|
|
|
|
|
"""
|
|
|
|
|
return await PluginInfo.filter(load_status=True).values_list(*args)
|
2024-08-24 20:24:05 +08:00
|
|
|
|
2024-08-20 21:29:42 +08:00
|
|
|
@classmethod
|
|
|
|
|
async def get_plugins_info(cls) -> BuildImage | str:
|
|
|
|
|
"""插件列表
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
BuildImage | str: 返回消息
|
|
|
|
|
"""
|
2024-09-02 01:50:39 +08:00
|
|
|
data: dict[str, StorePluginInfo] = await cls.__get_data()
|
2024-08-20 21:29:42 +08:00
|
|
|
column_name = ["-", "ID", "名称", "简介", "作者", "版本", "类型"]
|
2024-09-02 12:00:27 +08:00
|
|
|
plugin_list = await cls.get_loaded_plugins("module", "version")
|
2024-08-30 23:50:45 +08:00
|
|
|
suc_plugin = {p[0]: (p[1] or "0.1") for p in plugin_list}
|
2024-08-20 21:29:42 +08:00
|
|
|
data_list = [
|
|
|
|
|
[
|
2024-09-02 01:50:39 +08:00
|
|
|
"已安装" if plugin_info[1].module in suc_plugin else "",
|
2024-08-24 14:49:55 +08:00
|
|
|
id,
|
|
|
|
|
plugin_info[0],
|
2024-09-02 01:50:39 +08:00
|
|
|
plugin_info[1].description,
|
|
|
|
|
plugin_info[1].author,
|
2024-08-24 20:24:05 +08:00
|
|
|
cls.version_check(plugin_info[1], suc_plugin),
|
2024-09-02 01:50:39 +08:00
|
|
|
plugin_info[1].plugin_type_name,
|
2024-08-20 21:29:42 +08:00
|
|
|
]
|
2024-08-24 14:49:55 +08:00
|
|
|
for id, plugin_info in enumerate(data.items())
|
2024-08-20 21:29:42 +08:00
|
|
|
]
|
|
|
|
|
return await ImageTemplate.table_page(
|
|
|
|
|
"插件列表",
|
2024-08-30 23:50:45 +08:00
|
|
|
"通过添加/移除插件 ID 来管理插件",
|
2024-08-20 21:29:42 +08:00
|
|
|
column_name,
|
|
|
|
|
data_list,
|
|
|
|
|
text_style=row_style,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
async def add_plugin(cls, plugin_id: int) -> str:
|
2024-08-24 20:24:05 +08:00
|
|
|
"""添加插件
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
plugin_id: 插件id
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
str: 返回消息
|
|
|
|
|
"""
|
2024-09-02 01:50:39 +08:00
|
|
|
data: dict[str, StorePluginInfo] = await cls.__get_data()
|
2024-08-20 21:29:42 +08:00
|
|
|
if plugin_id < 0 or plugin_id >= len(data):
|
|
|
|
|
return "插件ID不存在..."
|
|
|
|
|
plugin_key = list(data.keys())[plugin_id]
|
2024-09-02 15:00:37 +08:00
|
|
|
plugin_list = await cls.get_loaded_plugins("module")
|
2024-08-20 21:29:42 +08:00
|
|
|
plugin_info = data[plugin_key]
|
2024-09-02 15:00:37 +08:00
|
|
|
if plugin_info.module in [p[0] for p in plugin_list]:
|
|
|
|
|
return f"插件 {plugin_key} 已安装,无需重复安装"
|
2024-09-02 01:50:39 +08:00
|
|
|
is_external = True
|
|
|
|
|
if plugin_info.github_url is None:
|
|
|
|
|
plugin_info.github_url = DEFAULT_GITHUB_URL
|
|
|
|
|
is_external = False
|
|
|
|
|
logger.info(f"正在安装插件 {plugin_key}...")
|
|
|
|
|
await cls.install_plugin_with_repo(
|
|
|
|
|
plugin_info.github_url,
|
|
|
|
|
plugin_info.module_path,
|
|
|
|
|
plugin_info.is_dir,
|
|
|
|
|
is_external,
|
|
|
|
|
)
|
|
|
|
|
return f"插件 {plugin_key} 安装成功! 重启后生效"
|
2024-08-26 10:39:33 +08:00
|
|
|
|
2024-09-02 01:50:39 +08:00
|
|
|
@classmethod
|
2024-09-03 13:04:49 +08:00
|
|
|
async def get_repo_package_info_of_jsd(cls, repo_info: RepoInfo) -> JsdPackageInfo:
|
2024-09-02 01:50:39 +08:00
|
|
|
"""获取插件包信息
|
2024-08-24 14:49:55 +08:00
|
|
|
|
2024-09-02 01:50:39 +08:00
|
|
|
参数:
|
|
|
|
|
repo_info: 仓库信息
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
JsdPackageInfo: 插件包信息
|
|
|
|
|
"""
|
|
|
|
|
jsd_package_url: str = JSD_PACKAGE_API_FORMAT.format(
|
|
|
|
|
owner=repo_info.owner, repo=repo_info.repo, branch=repo_info.branch
|
|
|
|
|
)
|
|
|
|
|
res = await AsyncHttpx.get(url=jsd_package_url)
|
|
|
|
|
if res.status_code != 200:
|
|
|
|
|
raise ValueError(f"下载错误, code: {res.status_code}")
|
|
|
|
|
return JsdPackageInfo(**res.json())
|
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
async def install_plugin_with_repo(
|
|
|
|
|
cls, github_url: str, module_path: str, is_dir: bool, is_external: bool = False
|
|
|
|
|
):
|
2024-09-03 13:04:49 +08:00
|
|
|
repo_info = RepoInfo.parse_github_url(github_url)
|
2024-09-02 01:50:39 +08:00
|
|
|
logger.debug(f"成功获取仓库信息: {repo_info}", "插件管理")
|
2024-09-03 13:04:49 +08:00
|
|
|
jsd_package_info: JsdPackageInfo = await cls.get_repo_package_info_of_jsd(
|
2024-09-02 01:50:39 +08:00
|
|
|
repo_info=repo_info
|
|
|
|
|
)
|
|
|
|
|
files = full_files_path(jsd_package_info, module_path, is_dir)
|
|
|
|
|
files = recurrence_files(
|
|
|
|
|
files,
|
|
|
|
|
module_path.replace(".", "/") + ("" if is_dir else ".py"),
|
|
|
|
|
is_dir,
|
|
|
|
|
)
|
|
|
|
|
logger.debug(f"获取插件文件列表: {files}", "插件管理")
|
2024-09-03 13:04:49 +08:00
|
|
|
download_urls = [
|
|
|
|
|
await repo_info.get_download_url_with_path(file) for file in files
|
|
|
|
|
]
|
2024-09-02 01:50:39 +08:00
|
|
|
base_path = BASE_PATH / "plugins" if is_external else BASE_PATH
|
|
|
|
|
download_paths: list[Path | str] = [base_path / file for file in files]
|
|
|
|
|
logger.debug(f"插件下载路径: {download_paths}", "插件管理")
|
|
|
|
|
result = await AsyncHttpx.gather_download_file(download_urls, download_paths)
|
|
|
|
|
for _id, success in enumerate(result):
|
|
|
|
|
if not success:
|
|
|
|
|
break
|
|
|
|
|
else:
|
|
|
|
|
# 安装依赖
|
|
|
|
|
plugin_path = base_path / "/".join(module_path.split("."))
|
|
|
|
|
req_files = recurrence_files(
|
|
|
|
|
jsd_package_info.files, REQ_TXT_FILE_STRING, False
|
|
|
|
|
)
|
|
|
|
|
req_files.extend(
|
|
|
|
|
recurrence_files(jsd_package_info.files, "requirement.txt", False)
|
|
|
|
|
)
|
|
|
|
|
logger.debug(f"获取插件依赖文件列表: {req_files}", "插件管理")
|
|
|
|
|
req_download_urls = [
|
2024-09-03 13:04:49 +08:00
|
|
|
await repo_info.get_download_url_with_path(file) for file in req_files
|
2024-09-02 01:50:39 +08:00
|
|
|
]
|
|
|
|
|
req_paths: list[Path | str] = [plugin_path / file for file in req_files]
|
|
|
|
|
logger.debug(f"插件依赖文件下载路径: {req_paths}", "插件管理")
|
|
|
|
|
if req_files:
|
|
|
|
|
result = await AsyncHttpx.gather_download_file(
|
|
|
|
|
req_download_urls, req_paths
|
|
|
|
|
)
|
|
|
|
|
for _id, success in enumerate(result):
|
|
|
|
|
if not success:
|
|
|
|
|
break
|
|
|
|
|
else:
|
|
|
|
|
logger.debug(f"插件依赖文件列表: {req_paths}", "插件管理")
|
|
|
|
|
install_requirement(plugin_path)
|
|
|
|
|
raise Exception("插件依赖文件下载失败")
|
|
|
|
|
return True
|
|
|
|
|
raise Exception("插件下载失败")
|
2024-08-20 21:29:42 +08:00
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
async def remove_plugin(cls, plugin_id: int) -> str:
|
2024-08-24 20:24:05 +08:00
|
|
|
"""移除插件
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
plugin_id: 插件id
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
str: 返回消息
|
|
|
|
|
"""
|
2024-09-02 01:50:39 +08:00
|
|
|
data = await cls.__get_data()
|
2024-08-20 21:29:42 +08:00
|
|
|
if plugin_id < 0 or plugin_id >= len(data):
|
|
|
|
|
return "插件ID不存在..."
|
|
|
|
|
plugin_key = list(data.keys())[plugin_id]
|
|
|
|
|
plugin_info = data[plugin_key]
|
|
|
|
|
path = BASE_PATH
|
2024-09-02 01:50:39 +08:00
|
|
|
if plugin_info.github_url:
|
2024-08-26 19:43:12 +08:00
|
|
|
path = BASE_PATH / "plugins"
|
2024-09-02 01:50:39 +08:00
|
|
|
for p in plugin_info.module_path.split("."):
|
2024-08-20 21:29:42 +08:00
|
|
|
path = path / p
|
2024-09-02 01:50:39 +08:00
|
|
|
if not plugin_info.is_dir:
|
2024-08-20 21:29:42 +08:00
|
|
|
path = Path(f"{path}.py")
|
|
|
|
|
if not path.exists():
|
|
|
|
|
return f"插件 {plugin_key} 不存在..."
|
|
|
|
|
logger.debug(f"尝试移除插件 {plugin_key} 文件: {path}", "插件管理")
|
2024-09-02 01:50:39 +08:00
|
|
|
if plugin_info.is_dir:
|
2024-08-20 21:29:42 +08:00
|
|
|
shutil.rmtree(path)
|
|
|
|
|
else:
|
|
|
|
|
path.unlink()
|
2024-08-26 10:39:33 +08:00
|
|
|
return f"插件 {plugin_key} 移除成功! 重启后生效"
|
2024-08-24 20:24:05 +08:00
|
|
|
|
2024-08-24 14:49:55 +08:00
|
|
|
@classmethod
|
|
|
|
|
async def search_plugin(cls, plugin_name_or_author: str) -> BuildImage | str:
|
2024-08-24 20:24:05 +08:00
|
|
|
"""搜索插件
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
plugin_name_or_author: 插件名称或作者
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
BuildImage | str: 返回消息
|
|
|
|
|
"""
|
2024-09-02 01:50:39 +08:00
|
|
|
data = await cls.__get_data()
|
2024-09-02 12:00:27 +08:00
|
|
|
plugin_list = await cls.get_loaded_plugins("module", "version")
|
2024-09-02 15:00:37 +08:00
|
|
|
suc_plugin = {p[0]: (p[1] or "Unknown") for p in plugin_list}
|
2024-08-24 14:49:55 +08:00
|
|
|
filtered_data = [
|
|
|
|
|
(id, plugin_info)
|
|
|
|
|
for id, plugin_info in enumerate(data.items())
|
2024-08-24 20:24:05 +08:00
|
|
|
if plugin_name_or_author.lower() in plugin_info[0].lower()
|
2024-09-02 01:50:39 +08:00
|
|
|
or plugin_name_or_author.lower() in plugin_info[1].author.lower()
|
2024-08-24 14:49:55 +08:00
|
|
|
]
|
|
|
|
|
|
|
|
|
|
data_list = [
|
|
|
|
|
[
|
2024-09-02 01:50:39 +08:00
|
|
|
"已安装" if plugin_info[1].module in suc_plugin else "",
|
2024-08-24 14:49:55 +08:00
|
|
|
id,
|
|
|
|
|
plugin_info[0],
|
2024-09-02 01:50:39 +08:00
|
|
|
plugin_info[1].description,
|
|
|
|
|
plugin_info[1].author,
|
2024-08-24 20:24:05 +08:00
|
|
|
cls.version_check(plugin_info[1], suc_plugin),
|
2024-09-02 01:50:39 +08:00
|
|
|
plugin_info[1].plugin_type_name,
|
2024-08-24 14:49:55 +08:00
|
|
|
]
|
|
|
|
|
for id, plugin_info in filtered_data
|
|
|
|
|
]
|
2024-08-24 20:00:29 +08:00
|
|
|
if not data_list:
|
2024-08-24 19:07:06 +08:00
|
|
|
return "未找到相关插件..."
|
2024-08-29 22:01:34 +08:00
|
|
|
column_name = ["-", "ID", "名称", "简介", "作者", "版本", "类型"]
|
2024-08-24 14:49:55 +08:00
|
|
|
return await ImageTemplate.table_page(
|
|
|
|
|
"插件列表",
|
2024-08-29 22:01:34 +08:00
|
|
|
"通过添加/移除插件 ID 来管理插件",
|
2024-08-24 14:49:55 +08:00
|
|
|
column_name,
|
|
|
|
|
data_list,
|
|
|
|
|
text_style=row_style,
|
2024-08-24 19:07:06 +08:00
|
|
|
)
|
2024-08-24 20:24:05 +08:00
|
|
|
|
2024-08-24 19:07:06 +08:00
|
|
|
@classmethod
|
|
|
|
|
async def update_plugin(cls, plugin_id: int) -> str:
|
2024-08-24 20:24:05 +08:00
|
|
|
"""更新插件
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
plugin_id: 插件id
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
str: 返回消息
|
|
|
|
|
"""
|
2024-09-02 01:50:39 +08:00
|
|
|
data = await cls.__get_data()
|
2024-08-24 19:07:06 +08:00
|
|
|
if plugin_id < 0 or plugin_id >= len(data):
|
|
|
|
|
return "插件ID不存在..."
|
|
|
|
|
plugin_key = list(data.keys())[plugin_id]
|
2024-09-02 01:50:39 +08:00
|
|
|
logger.info(f"尝试更新插件 {plugin_key}", "插件管理")
|
2024-08-24 19:07:06 +08:00
|
|
|
plugin_info = data[plugin_key]
|
2024-09-02 12:00:27 +08:00
|
|
|
plugin_list = await cls.get_loaded_plugins("module", "version")
|
2024-09-02 15:00:37 +08:00
|
|
|
suc_plugin = {p[0]: (p[1] or "Unknown") for p in plugin_list}
|
|
|
|
|
if plugin_info.module not in [p[0] for p in plugin_list]:
|
|
|
|
|
return f"插件 {plugin_key} 未安装,无法更新"
|
2024-09-02 01:50:39 +08:00
|
|
|
logger.debug(f"当前插件列表: {suc_plugin}", "插件管理")
|
|
|
|
|
if cls.check_version_is_new(plugin_info, suc_plugin):
|
|
|
|
|
return f"插件 {plugin_key} 已是最新版本"
|
|
|
|
|
is_external = True
|
|
|
|
|
if plugin_info.github_url is None:
|
|
|
|
|
plugin_info.github_url = DEFAULT_GITHUB_URL
|
|
|
|
|
is_external = False
|
|
|
|
|
await cls.install_plugin_with_repo(
|
|
|
|
|
plugin_info.github_url,
|
|
|
|
|
plugin_info.module_path,
|
|
|
|
|
plugin_info.is_dir,
|
|
|
|
|
is_external,
|
|
|
|
|
)
|
2024-08-26 10:39:33 +08:00
|
|
|
return f"插件 {plugin_key} 更新成功! 重启后生效"
|