2024-12-10 19:49:11 +08:00
|
|
|
|
from pathlib import Path
|
2025-08-05 17:49:23 +08:00
|
|
|
|
import random
|
2024-08-20 21:29:42 +08:00
|
|
|
|
import shutil
|
|
|
|
|
|
|
2024-09-03 13:04:49 +08:00
|
|
|
|
from aiocache import cached
|
2024-12-10 19:49:11 +08:00
|
|
|
|
import ujson as json
|
2024-08-20 21:29:42 +08:00
|
|
|
|
|
2024-12-10 19:49:11 +08:00
|
|
|
|
from zhenxun.builtin_plugins.plugin_store.models import StorePluginInfo
|
2025-08-05 17:49:23 +08:00
|
|
|
|
from zhenxun.configs.path_config import TEMP_PATH
|
2024-08-29 22:01:34 +08:00
|
|
|
|
from zhenxun.models.plugin_info import PluginInfo
|
2024-12-10 19:49:11 +08:00
|
|
|
|
from zhenxun.services.log import logger
|
|
|
|
|
|
from zhenxun.services.plugin_init import PluginInitManager
|
|
|
|
|
|
from zhenxun.utils.image_utils import BuildImage, ImageTemplate, RowStyle
|
2025-06-20 19:08:06 +08:00
|
|
|
|
from zhenxun.utils.manager.virtual_env_package_manager import VirtualEnvPackageManager
|
2025-08-05 17:49:23 +08:00
|
|
|
|
from zhenxun.utils.repo_utils import RepoFileManager
|
|
|
|
|
|
from zhenxun.utils.repo_utils.models import RepoFileInfo, RepoType
|
2025-01-09 10:29:49 +08:00
|
|
|
|
from zhenxun.utils.utils import is_number
|
2024-08-20 21:29:42 +08:00
|
|
|
|
|
2025-06-20 19:08:06 +08:00
|
|
|
|
from .config import (
|
|
|
|
|
|
BASE_PATH,
|
|
|
|
|
|
DEFAULT_GITHUB_URL,
|
|
|
|
|
|
EXTRA_GITHUB_URL,
|
|
|
|
|
|
LOG_COMMAND,
|
|
|
|
|
|
)
|
2025-08-05 17:49:23 +08:00
|
|
|
|
from .exceptions import PluginStoreException
|
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
|
|
|
|
|
|
|
|
|
|
|
|
|
2025-06-20 19:08:06 +08:00
|
|
|
|
class StoreManager:
|
2024-08-20 21:29:42 +08:00
|
|
|
|
@classmethod
|
2024-09-03 13:04:49 +08:00
|
|
|
|
@cached(60)
|
2025-08-05 17:49:23 +08:00
|
|
|
|
async def get_data(cls) -> tuple[list[StorePluginInfo], list[StorePluginInfo]]:
|
2024-08-20 21:29:42 +08:00
|
|
|
|
"""获取插件信息数据
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
2025-08-05 17:49:23 +08:00
|
|
|
|
tuple[list[StorePluginInfo], list[StorePluginInfo]]:
|
|
|
|
|
|
原生插件信息数据,第三方插件信息数据
|
2024-08-20 21:29:42 +08:00
|
|
|
|
"""
|
2025-08-05 17:49:23 +08:00
|
|
|
|
plugins = await RepoFileManager.get_file_content(
|
|
|
|
|
|
DEFAULT_GITHUB_URL, "plugins.json"
|
|
|
|
|
|
)
|
|
|
|
|
|
extra_plugins = await RepoFileManager.get_file_content(
|
|
|
|
|
|
EXTRA_GITHUB_URL, "plugins.json", "index"
|
|
|
|
|
|
)
|
|
|
|
|
|
return [StorePluginInfo(**plugin) for plugin in json.loads(plugins)], [
|
|
|
|
|
|
StorePluginInfo(**plugin) for plugin in json.loads(extra_plugins)
|
|
|
|
|
|
]
|
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
|
2025-06-20 19:08:06 +08:00
|
|
|
|
suc_plugin: 模块名: 版本号
|
2024-09-02 01:50:39 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
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
|
2025-06-20 19:08:06 +08:00
|
|
|
|
suc_plugin: 模块名: 版本号
|
2024-09-02 01:50:39 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
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
|
2025-08-05 17:49:23 +08:00
|
|
|
|
async def get_plugins_info(cls) -> list[BuildImage] | str:
|
2024-08-20 21:29:42 +08:00
|
|
|
|
"""插件列表
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
BuildImage | str: 返回消息
|
|
|
|
|
|
"""
|
2025-08-05 17:49:23 +08:00
|
|
|
|
plugin_list, extra_plugin_list = await cls.get_data()
|
2024-08-20 21:29:42 +08:00
|
|
|
|
column_name = ["-", "ID", "名称", "简介", "作者", "版本", "类型"]
|
2025-06-20 19:08:06 +08:00
|
|
|
|
db_plugin_list = await cls.get_loaded_plugins("module", "version")
|
|
|
|
|
|
suc_plugin = {p[0]: (p[1] or "0.1") for p in db_plugin_list}
|
2025-08-05 17:49:23 +08:00
|
|
|
|
index = 0
|
|
|
|
|
|
data_list = []
|
|
|
|
|
|
extra_data_list = []
|
|
|
|
|
|
for plugin_info in plugin_list:
|
|
|
|
|
|
data_list.append(
|
|
|
|
|
|
[
|
|
|
|
|
|
"已安装" if plugin_info.module in suc_plugin else "",
|
|
|
|
|
|
index,
|
|
|
|
|
|
plugin_info.name,
|
|
|
|
|
|
plugin_info.description,
|
|
|
|
|
|
plugin_info.author,
|
|
|
|
|
|
cls.version_check(plugin_info, suc_plugin),
|
|
|
|
|
|
plugin_info.plugin_type_name,
|
|
|
|
|
|
]
|
|
|
|
|
|
)
|
|
|
|
|
|
index += 1
|
|
|
|
|
|
for plugin_info in extra_plugin_list:
|
|
|
|
|
|
extra_data_list.append(
|
|
|
|
|
|
[
|
|
|
|
|
|
"已安装" if plugin_info.module in suc_plugin else "",
|
|
|
|
|
|
index,
|
|
|
|
|
|
plugin_info.name,
|
|
|
|
|
|
plugin_info.description,
|
|
|
|
|
|
plugin_info.author,
|
|
|
|
|
|
cls.version_check(plugin_info, suc_plugin),
|
|
|
|
|
|
plugin_info.plugin_type_name,
|
|
|
|
|
|
]
|
|
|
|
|
|
)
|
|
|
|
|
|
index += 1
|
|
|
|
|
|
return [
|
|
|
|
|
|
await ImageTemplate.table_page(
|
|
|
|
|
|
"原生插件列表",
|
|
|
|
|
|
"通过添加/移除插件 ID 来管理插件",
|
|
|
|
|
|
column_name,
|
|
|
|
|
|
data_list,
|
|
|
|
|
|
text_style=row_style,
|
|
|
|
|
|
),
|
|
|
|
|
|
await ImageTemplate.table_page(
|
|
|
|
|
|
"第三方插件列表",
|
|
|
|
|
|
"通过添加/移除插件 ID 来管理插件",
|
|
|
|
|
|
column_name,
|
|
|
|
|
|
extra_data_list,
|
|
|
|
|
|
text_style=row_style,
|
|
|
|
|
|
),
|
2024-08-20 21:29:42 +08:00
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
@classmethod
|
2025-08-05 17:49:23 +08:00
|
|
|
|
async def get_plugin_by_value(
|
|
|
|
|
|
cls, index_or_module: str, is_update: bool = False
|
|
|
|
|
|
) -> tuple[StorePluginInfo, bool]:
|
|
|
|
|
|
"""获取插件信息
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
index_or_module: 插件索引或模块名
|
|
|
|
|
|
is_update: 是否是更新插件
|
|
|
|
|
|
|
|
|
|
|
|
异常:
|
|
|
|
|
|
PluginStoreException: 插件不存在
|
|
|
|
|
|
PluginStoreException: 插件已安装
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
StorePluginInfo: 插件信息
|
|
|
|
|
|
bool: 是否是外部插件
|
|
|
|
|
|
"""
|
|
|
|
|
|
plugin_list, extra_plugin_list = await cls.get_data()
|
|
|
|
|
|
plugin_info = None
|
|
|
|
|
|
is_external = False
|
|
|
|
|
|
db_plugin_list = await cls.get_loaded_plugins("module")
|
|
|
|
|
|
plugin_key = await cls._resolve_plugin_key(index_or_module)
|
|
|
|
|
|
for p in plugin_list:
|
|
|
|
|
|
if p.module == plugin_key:
|
|
|
|
|
|
is_external = False
|
|
|
|
|
|
plugin_info = p
|
|
|
|
|
|
break
|
|
|
|
|
|
for p in extra_plugin_list:
|
|
|
|
|
|
if p.module == plugin_key:
|
|
|
|
|
|
is_external = True
|
|
|
|
|
|
plugin_info = p
|
|
|
|
|
|
break
|
|
|
|
|
|
if not plugin_info:
|
|
|
|
|
|
raise PluginStoreException(f"插件不存在: {plugin_key}")
|
|
|
|
|
|
if not is_update and plugin_info.module in [p[0] for p in db_plugin_list]:
|
|
|
|
|
|
raise PluginStoreException(f"插件 {plugin_info.name} 已安装,无需重复安装")
|
|
|
|
|
|
if plugin_info.module not in [p[0] for p in db_plugin_list] and is_update:
|
|
|
|
|
|
raise PluginStoreException(f"插件 {plugin_info.name} 未安装,无法更新")
|
|
|
|
|
|
return plugin_info, is_external
|
|
|
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
|
async def add_plugin(cls, index_or_module: str) -> str:
|
2024-08-24 20:24:05 +08:00
|
|
|
|
"""添加插件
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
2024-10-02 18:32:21 +08:00
|
|
|
|
plugin_id: 插件id或模块名
|
2024-08-24 20:24:05 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
str: 返回消息
|
|
|
|
|
|
"""
|
2025-08-05 17:49:23 +08:00
|
|
|
|
plugin_info, is_external = await cls.get_plugin_by_value(index_or_module)
|
2024-09-02 01:50:39 +08:00
|
|
|
|
if plugin_info.github_url is None:
|
|
|
|
|
|
plugin_info.github_url = DEFAULT_GITHUB_URL
|
2024-09-21 17:38:37 +08:00
|
|
|
|
version_split = plugin_info.version.split("-")
|
|
|
|
|
|
if len(version_split) > 1:
|
|
|
|
|
|
github_url_split = plugin_info.github_url.split("/tree/")
|
|
|
|
|
|
plugin_info.github_url = f"{github_url_split[0]}/tree/{version_split[1]}"
|
2025-06-20 19:08:06 +08:00
|
|
|
|
logger.info(f"正在安装插件 {plugin_info.name}...", LOG_COMMAND)
|
2024-09-02 01:50:39 +08:00
|
|
|
|
await cls.install_plugin_with_repo(
|
|
|
|
|
|
plugin_info.github_url,
|
|
|
|
|
|
plugin_info.module_path,
|
|
|
|
|
|
plugin_info.is_dir,
|
|
|
|
|
|
is_external,
|
|
|
|
|
|
)
|
2025-06-20 19:08:06 +08:00
|
|
|
|
return f"插件 {plugin_info.name} 安装成功! 重启后生效"
|
2024-08-26 10:39:33 +08:00
|
|
|
|
|
2024-09-02 01:50:39 +08:00
|
|
|
|
@classmethod
|
|
|
|
|
|
async def install_plugin_with_repo(
|
2025-06-20 19:08:06 +08:00
|
|
|
|
cls,
|
|
|
|
|
|
github_url: str,
|
|
|
|
|
|
module_path: str,
|
|
|
|
|
|
is_dir: bool,
|
|
|
|
|
|
is_external: bool = False,
|
2024-09-02 01:50:39 +08:00
|
|
|
|
):
|
2025-08-05 17:49:23 +08:00
|
|
|
|
"""安装插件
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
github_url: 仓库地址
|
|
|
|
|
|
module_path: 模块路径
|
|
|
|
|
|
is_dir: 是否是文件夹
|
|
|
|
|
|
is_external: 是否是外部仓库
|
|
|
|
|
|
"""
|
|
|
|
|
|
repo_type = RepoType.GITHUB if is_external else None
|
2025-02-03 21:23:14 +08:00
|
|
|
|
replace_module_path = module_path.replace(".", "/")
|
2025-08-05 17:49:23 +08:00
|
|
|
|
if is_dir:
|
|
|
|
|
|
files = await RepoFileManager.list_directory_files(
|
|
|
|
|
|
github_url, replace_module_path, repo_type=repo_type
|
|
|
|
|
|
)
|
2024-09-02 01:50:39 +08:00
|
|
|
|
else:
|
2025-08-05 17:49:23 +08:00
|
|
|
|
files = [RepoFileInfo(path=f"{replace_module_path}.py", is_dir=False)]
|
|
|
|
|
|
local_path = BASE_PATH / "plugins" if is_external else BASE_PATH
|
|
|
|
|
|
files = [file for file in files if not file.is_dir]
|
|
|
|
|
|
download_files = [(file.path, local_path / file.path) for file in files]
|
|
|
|
|
|
await RepoFileManager.download_files(
|
|
|
|
|
|
github_url, download_files, repo_type=repo_type
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
requirement_paths = [
|
|
|
|
|
|
file
|
|
|
|
|
|
for file in files
|
|
|
|
|
|
if file.path.endswith("requirement.txt")
|
|
|
|
|
|
or file.path.endswith("requirements.txt")
|
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
is_install_req = False
|
|
|
|
|
|
for requirement_path in requirement_paths:
|
|
|
|
|
|
requirement_file = local_path / requirement_path.path
|
|
|
|
|
|
if requirement_file.exists():
|
|
|
|
|
|
is_install_req = True
|
|
|
|
|
|
await VirtualEnvPackageManager.install_requirement(requirement_file)
|
|
|
|
|
|
|
|
|
|
|
|
if not is_install_req:
|
|
|
|
|
|
# 从仓库根目录查找文件
|
|
|
|
|
|
rand = random.randint(1, 10000)
|
|
|
|
|
|
requirement_path = TEMP_PATH / f"plugin_store_{rand}_req.txt"
|
|
|
|
|
|
requirements_path = TEMP_PATH / f"plugin_store_{rand}_reqs.txt"
|
|
|
|
|
|
await RepoFileManager.download_files(
|
|
|
|
|
|
github_url,
|
|
|
|
|
|
[
|
|
|
|
|
|
("requirement.txt", requirement_path),
|
|
|
|
|
|
("requirements.txt", requirements_path),
|
|
|
|
|
|
],
|
|
|
|
|
|
repo_type=repo_type,
|
|
|
|
|
|
ignore_error=True,
|
|
|
|
|
|
)
|
|
|
|
|
|
if requirement_path.exists():
|
|
|
|
|
|
logger.info(
|
|
|
|
|
|
f"开始安装插件 {module_path} 依赖文件: {requirement_path}",
|
|
|
|
|
|
LOG_COMMAND,
|
2025-02-03 21:23:14 +08:00
|
|
|
|
)
|
2025-08-05 17:49:23 +08:00
|
|
|
|
await VirtualEnvPackageManager.install_requirement(requirement_path)
|
|
|
|
|
|
if requirements_path.exists():
|
|
|
|
|
|
logger.info(
|
|
|
|
|
|
f"开始安装插件 {module_path} 依赖文件: {requirements_path}",
|
|
|
|
|
|
LOG_COMMAND,
|
2024-09-02 01:50:39 +08:00
|
|
|
|
)
|
2025-08-05 17:49:23 +08:00
|
|
|
|
await VirtualEnvPackageManager.install_requirement(requirements_path)
|
2024-08-20 21:29:42 +08:00
|
|
|
|
|
|
|
|
|
|
@classmethod
|
2025-08-05 17:49:23 +08:00
|
|
|
|
async def remove_plugin(cls, index_or_module: str) -> str:
|
2024-08-24 20:24:05 +08:00
|
|
|
|
"""移除插件
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
2025-08-05 17:49:23 +08:00
|
|
|
|
index_or_module: 插件id或模块名
|
2024-08-24 20:24:05 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
str: 返回消息
|
|
|
|
|
|
"""
|
2025-08-05 17:49:23 +08:00
|
|
|
|
plugin_info, _ = await cls.get_plugin_by_value(index_or_module)
|
2024-08-20 21:29:42 +08:00
|
|
|
|
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():
|
2025-06-20 19:08:06 +08:00
|
|
|
|
return f"插件 {plugin_info.name} 不存在..."
|
|
|
|
|
|
logger.debug(f"尝试移除插件 {plugin_info.name} 文件: {path}", LOG_COMMAND)
|
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-09-27 10:29:59 +08:00
|
|
|
|
await PluginInitManager.remove(f"zhenxun.{plugin_info.module_path}")
|
2025-06-20 19:08:06 +08:00
|
|
|
|
return f"插件 {plugin_info.name} 移除成功! 重启后生效"
|
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: 返回消息
|
|
|
|
|
|
"""
|
2025-08-05 17:49:23 +08:00
|
|
|
|
plugin_list, extra_plugin_list = await cls.get_data()
|
|
|
|
|
|
all_plugin_list = plugin_list + extra_plugin_list
|
2025-06-20 19:08:06 +08:00
|
|
|
|
db_plugin_list = await cls.get_loaded_plugins("module", "version")
|
|
|
|
|
|
suc_plugin = {p[0]: (p[1] or "Unknown") for p in db_plugin_list}
|
2024-08-24 14:49:55 +08:00
|
|
|
|
filtered_data = [
|
|
|
|
|
|
(id, plugin_info)
|
2025-08-05 17:49:23 +08:00
|
|
|
|
for id, plugin_info in enumerate(all_plugin_list)
|
2025-06-20 19:08:06 +08:00
|
|
|
|
if plugin_name_or_author.lower() in plugin_info.name.lower()
|
|
|
|
|
|
or plugin_name_or_author.lower() in plugin_info.author.lower()
|
2024-08-24 14:49:55 +08:00
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
data_list = [
|
|
|
|
|
|
[
|
2025-06-20 19:08:06 +08:00
|
|
|
|
"已安装" if plugin_info.module in suc_plugin else "",
|
2024-08-24 14:49:55 +08:00
|
|
|
|
id,
|
2025-06-20 19:08:06 +08:00
|
|
|
|
plugin_info.name,
|
|
|
|
|
|
plugin_info.description,
|
|
|
|
|
|
plugin_info.author,
|
|
|
|
|
|
cls.version_check(plugin_info, suc_plugin),
|
|
|
|
|
|
plugin_info.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(
|
2025-06-20 19:08:06 +08:00
|
|
|
|
"商店插件列表",
|
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
|
2025-08-05 17:49:23 +08:00
|
|
|
|
async def update_plugin(cls, index_or_module: str) -> str:
|
2024-08-24 20:24:05 +08:00
|
|
|
|
"""更新插件
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
2025-08-05 17:49:23 +08:00
|
|
|
|
index_or_module: 插件id
|
2024-08-24 20:24:05 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
str: 返回消息
|
|
|
|
|
|
"""
|
2025-08-05 17:49:23 +08:00
|
|
|
|
plugin_info, is_external = await cls.get_plugin_by_value(index_or_module, True)
|
2025-06-20 19:08:06 +08:00
|
|
|
|
logger.info(f"尝试更新插件 {plugin_info.name}", LOG_COMMAND)
|
|
|
|
|
|
db_plugin_list = await cls.get_loaded_plugins("module", "version")
|
|
|
|
|
|
suc_plugin = {p[0]: (p[1] or "Unknown") for p in db_plugin_list}
|
|
|
|
|
|
logger.debug(f"当前插件列表: {suc_plugin}", LOG_COMMAND)
|
2024-09-02 01:50:39 +08:00
|
|
|
|
if cls.check_version_is_new(plugin_info, suc_plugin):
|
2025-06-20 19:08:06 +08:00
|
|
|
|
return f"插件 {plugin_info.name} 已是最新版本"
|
2024-09-02 01:50:39 +08:00
|
|
|
|
if plugin_info.github_url is None:
|
|
|
|
|
|
plugin_info.github_url = DEFAULT_GITHUB_URL
|
|
|
|
|
|
await cls.install_plugin_with_repo(
|
|
|
|
|
|
plugin_info.github_url,
|
|
|
|
|
|
plugin_info.module_path,
|
|
|
|
|
|
plugin_info.is_dir,
|
|
|
|
|
|
is_external,
|
|
|
|
|
|
)
|
2025-06-20 19:08:06 +08:00
|
|
|
|
return f"插件 {plugin_info.name} 更新成功! 重启后生效"
|
2024-10-02 18:32:21 +08:00
|
|
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
|
|
async def update_all_plugin(cls) -> str:
|
|
|
|
|
|
"""更新插件
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
plugin_id: 插件id
|
|
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
str: 返回消息
|
|
|
|
|
|
"""
|
2025-08-05 17:49:23 +08:00
|
|
|
|
plugin_list, extra_plugin_list = await cls.get_data()
|
|
|
|
|
|
all_plugin_list = plugin_list + extra_plugin_list
|
|
|
|
|
|
plugin_name_list = [p.name for p in all_plugin_list]
|
2024-12-19 09:15:39 +08:00
|
|
|
|
update_failed_list = []
|
|
|
|
|
|
update_success_list = []
|
|
|
|
|
|
result = "--已更新{}个插件 {}个失败 {}个成功--"
|
2025-06-20 19:08:06 +08:00
|
|
|
|
logger.info(f"尝试更新全部插件 {plugin_name_list}", LOG_COMMAND)
|
|
|
|
|
|
for plugin_info in plugin_list:
|
2024-12-19 09:15:39 +08:00
|
|
|
|
try:
|
2025-06-20 19:08:06 +08:00
|
|
|
|
db_plugin_list = await cls.get_loaded_plugins("module", "version")
|
|
|
|
|
|
suc_plugin = {p[0]: (p[1] or "Unknown") for p in db_plugin_list}
|
|
|
|
|
|
if plugin_info.module not in [p[0] for p in db_plugin_list]:
|
|
|
|
|
|
logger.debug(
|
|
|
|
|
|
f"插件 {plugin_info.name}({plugin_info.module}) 未安装,跳过",
|
|
|
|
|
|
LOG_COMMAND,
|
|
|
|
|
|
)
|
2024-12-19 09:15:39 +08:00
|
|
|
|
continue
|
|
|
|
|
|
if cls.check_version_is_new(plugin_info, suc_plugin):
|
2025-06-20 19:08:06 +08:00
|
|
|
|
logger.debug(
|
2025-06-23 19:29:03 +08:00
|
|
|
|
f"插件 {plugin_info.name}({plugin_info.module}) "
|
|
|
|
|
|
"已是最新版本,跳过",
|
2025-06-20 19:08:06 +08:00
|
|
|
|
LOG_COMMAND,
|
|
|
|
|
|
)
|
2024-12-19 09:15:39 +08:00
|
|
|
|
continue
|
2025-06-20 19:08:06 +08:00
|
|
|
|
logger.info(
|
|
|
|
|
|
f"正在更新插件 {plugin_info.name}({plugin_info.module})",
|
|
|
|
|
|
LOG_COMMAND,
|
|
|
|
|
|
)
|
2024-12-19 09:15:39 +08:00
|
|
|
|
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,
|
|
|
|
|
|
)
|
2025-06-20 19:08:06 +08:00
|
|
|
|
update_success_list.append(plugin_info.name)
|
2024-12-19 09:15:39 +08:00
|
|
|
|
except Exception as e:
|
2025-06-20 19:08:06 +08:00
|
|
|
|
logger.error(
|
|
|
|
|
|
f"更新插件 {plugin_info.name}({plugin_info.module}) 失败",
|
|
|
|
|
|
LOG_COMMAND,
|
|
|
|
|
|
e=e,
|
|
|
|
|
|
)
|
|
|
|
|
|
update_failed_list.append(plugin_info.name)
|
2024-12-19 09:15:39 +08:00
|
|
|
|
if not update_success_list and not update_failed_list:
|
2024-10-02 18:32:21 +08:00
|
|
|
|
return "全部插件已是最新版本"
|
2024-12-19 09:15:39 +08:00
|
|
|
|
if update_success_list:
|
|
|
|
|
|
result += "\n* 以下插件更新成功:\n\t- {}".format(
|
|
|
|
|
|
"\n\t- ".join(update_success_list)
|
|
|
|
|
|
)
|
|
|
|
|
|
if update_failed_list:
|
|
|
|
|
|
result += "\n* 以下插件更新失败:\n\t- {}".format(
|
|
|
|
|
|
"\n\t- ".join(update_failed_list)
|
|
|
|
|
|
)
|
|
|
|
|
|
return (
|
|
|
|
|
|
result.format(
|
|
|
|
|
|
len(update_success_list) + len(update_failed_list),
|
|
|
|
|
|
len(update_failed_list),
|
|
|
|
|
|
len(update_success_list),
|
|
|
|
|
|
)
|
|
|
|
|
|
+ "\n重启后生效"
|
2024-10-02 18:32:21 +08:00
|
|
|
|
)
|
2024-12-10 19:49:11 +08:00
|
|
|
|
|
2024-10-02 18:32:21 +08:00
|
|
|
|
@classmethod
|
2025-01-09 10:29:49 +08:00
|
|
|
|
async def _resolve_plugin_key(cls, plugin_id: str) -> str:
|
2025-06-23 19:29:03 +08:00
|
|
|
|
"""获取插件module
|
|
|
|
|
|
|
|
|
|
|
|
参数:
|
|
|
|
|
|
plugin_id: module,id或插件名称
|
|
|
|
|
|
|
|
|
|
|
|
异常:
|
2025-08-05 17:49:23 +08:00
|
|
|
|
PluginStoreException: 插件不存在
|
|
|
|
|
|
PluginStoreException: 插件不存在
|
2025-06-23 19:29:03 +08:00
|
|
|
|
|
|
|
|
|
|
返回:
|
|
|
|
|
|
str: 插件模块名
|
|
|
|
|
|
"""
|
2025-08-05 17:49:23 +08:00
|
|
|
|
plugin_list, extra_plugin_list = await cls.get_data()
|
|
|
|
|
|
all_plugin_list = plugin_list + extra_plugin_list
|
2025-01-09 10:29:49 +08:00
|
|
|
|
if is_number(plugin_id):
|
|
|
|
|
|
idx = int(plugin_id)
|
2025-08-05 17:49:23 +08:00
|
|
|
|
if idx < 0 or idx >= len(all_plugin_list):
|
|
|
|
|
|
raise PluginStoreException("插件ID不存在...")
|
|
|
|
|
|
return all_plugin_list[idx].module
|
2024-10-02 18:32:21 +08:00
|
|
|
|
elif isinstance(plugin_id, str):
|
2025-06-23 19:29:03 +08:00
|
|
|
|
result = (
|
2025-08-05 17:49:23 +08:00
|
|
|
|
None
|
|
|
|
|
|
if plugin_id not in [v.module for v in all_plugin_list]
|
|
|
|
|
|
else plugin_id
|
|
|
|
|
|
) or next(v for v in all_plugin_list if v.name == plugin_id).module
|
2025-06-23 19:29:03 +08:00
|
|
|
|
if not result:
|
2025-08-05 17:49:23 +08:00
|
|
|
|
raise PluginStoreException("插件 Module / 名称 不存在...")
|
2025-06-23 19:29:03 +08:00
|
|
|
|
return result
|