2024-03-21 01:20:18 +08:00
|
|
|
|
import json
|
|
|
|
|
import os.path
|
|
|
|
|
import shutil
|
2024-03-21 12:10:24 +08:00
|
|
|
|
import sys
|
|
|
|
|
from io import StringIO
|
2024-03-21 01:20:18 +08:00
|
|
|
|
from typing import Optional
|
|
|
|
|
|
2024-03-21 14:52:02 +08:00
|
|
|
|
import aiofiles
|
|
|
|
|
import aiohttp
|
2024-03-21 01:20:18 +08:00
|
|
|
|
import nonebot
|
2024-03-21 14:52:02 +08:00
|
|
|
|
import pip
|
2024-03-21 01:20:18 +08:00
|
|
|
|
from arclet.alconna import Arparma, MultiVar
|
|
|
|
|
from nonebot.permission import SUPERUSER
|
2024-03-21 14:52:02 +08:00
|
|
|
|
from nonebot_plugin_alconna import Alconna, Args, Subcommand, on_alconna
|
2024-03-21 01:20:18 +08:00
|
|
|
|
|
|
|
|
|
from src.utils.language import get_user_lang
|
2024-03-21 12:10:24 +08:00
|
|
|
|
from src.utils.message import Markdown as md, send_markdown
|
2024-03-21 01:20:18 +08:00
|
|
|
|
from src.utils.resource import get_res
|
|
|
|
|
from src.utils.typing import T_Bot, T_MessageEvent
|
2024-03-21 12:10:24 +08:00
|
|
|
|
from .common import *
|
2024-03-21 14:52:02 +08:00
|
|
|
|
from src.utils.data_manager import InstalledPlugin
|
2024-03-21 12:10:24 +08:00
|
|
|
|
|
2024-03-21 01:20:18 +08:00
|
|
|
|
npm_alc = on_alconna(
|
|
|
|
|
Alconna(
|
2024-03-21 14:52:02 +08:00
|
|
|
|
["lnpm", "插件管理"],
|
2024-03-21 01:20:18 +08:00
|
|
|
|
Subcommand(
|
|
|
|
|
"update",
|
|
|
|
|
alias=["u"],
|
|
|
|
|
),
|
|
|
|
|
Subcommand(
|
|
|
|
|
"search",
|
|
|
|
|
Args["keywords", MultiVar(str)]["page", int, 1],
|
2024-03-21 12:10:24 +08:00
|
|
|
|
alias=["s", "搜索"],
|
2024-03-21 01:20:18 +08:00
|
|
|
|
),
|
|
|
|
|
Subcommand(
|
|
|
|
|
"install",
|
|
|
|
|
Args["plugin_name", str],
|
2024-03-21 12:10:24 +08:00
|
|
|
|
alias=["i", "安装"],
|
2024-03-21 01:20:18 +08:00
|
|
|
|
),
|
|
|
|
|
Subcommand(
|
|
|
|
|
"remove",
|
|
|
|
|
Args["plugin_name", str],
|
2024-03-21 12:10:24 +08:00
|
|
|
|
alias=["rm", "移除", "卸载"],
|
2024-03-21 01:20:18 +08:00
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
permission=SUPERUSER
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class PluginTag(LiteModel):
|
|
|
|
|
label: str
|
|
|
|
|
color: str = '#000000'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class StorePlugin(LiteModel):
|
|
|
|
|
name: str
|
|
|
|
|
desc: str
|
|
|
|
|
module_name: str
|
|
|
|
|
project_link: str = ''
|
|
|
|
|
homepage: str = ''
|
|
|
|
|
author: str = ''
|
|
|
|
|
type: str | None = None
|
|
|
|
|
version: str | None = ''
|
|
|
|
|
time: str = ''
|
|
|
|
|
tags: list[PluginTag] = []
|
|
|
|
|
is_official: bool = False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@npm_alc.handle()
|
|
|
|
|
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|
|
|
|
ulang = get_user_lang(str(event.user_id))
|
|
|
|
|
|
|
|
|
|
if not os.path.exists("data/liteyuki/plugins.json"):
|
2024-03-21 01:52:25 +08:00
|
|
|
|
shutil.copy(get_res('unsorted/plugins.json'), "data/liteyuki/plugins.json")
|
2024-03-21 01:20:18 +08:00
|
|
|
|
nonebot.logger.info("Please update plugin store data file.")
|
|
|
|
|
|
|
|
|
|
if result.subcommands.get("update"):
|
|
|
|
|
r = await npm_update()
|
|
|
|
|
if r:
|
|
|
|
|
await npm_alc.finish(ulang.get("npm.store_update_success"))
|
|
|
|
|
else:
|
|
|
|
|
await npm_alc.finish(ulang.get("npm.store_update_failed"))
|
|
|
|
|
|
|
|
|
|
elif result.subcommands.get("search"):
|
|
|
|
|
keywords: list[str] = result.subcommands["search"].args.get("keywords")
|
|
|
|
|
rs = await npm_search(keywords)
|
2024-03-21 12:18:15 +08:00
|
|
|
|
max_show = 10
|
2024-03-21 01:20:18 +08:00
|
|
|
|
if len(rs):
|
|
|
|
|
reply = f"{ulang.get('npm.search_result')} | {ulang.get('npm.total', TOTAL=len(rs))}\n***"
|
2024-03-21 12:10:24 +08:00
|
|
|
|
for plugin in rs[:min(max_show, len(rs))]:
|
|
|
|
|
btn_install = md.button(ulang.get('npm.install'), 'lnpm install %s' % plugin.module_name)
|
|
|
|
|
link_page = md.link(ulang.get('npm.homepage'), plugin.homepage)
|
|
|
|
|
|
2024-03-21 14:52:02 +08:00
|
|
|
|
reply += (f"\n{btn_install} **{plugin.name}**\n"
|
2024-03-21 01:20:18 +08:00
|
|
|
|
f"\n > **{plugin.desc}**\n"
|
2024-03-21 14:52:02 +08:00
|
|
|
|
f"\n > {ulang.get('npm.author')}: {plugin.author} {link_page}\n\n***\n")
|
2024-03-21 12:10:24 +08:00
|
|
|
|
if len(rs) > max_show:
|
2024-03-21 14:52:02 +08:00
|
|
|
|
reply += f"\n{ulang.get('npm.too_many_results', HIDE_NUM=len(rs) - max_show)}"
|
2024-03-21 01:20:18 +08:00
|
|
|
|
else:
|
|
|
|
|
reply = ulang.get("npm.search_no_result")
|
|
|
|
|
await send_markdown(reply, bot, event=event)
|
|
|
|
|
|
2024-03-21 12:10:24 +08:00
|
|
|
|
elif result.subcommands.get("install"):
|
2024-03-21 14:52:02 +08:00
|
|
|
|
plugin_module_name: str = result.subcommands["install"].args.get("plugin_name")
|
|
|
|
|
await npm_alc.send(ulang.get("npm.installing", NAME=plugin_module_name))
|
|
|
|
|
r, log = npm_install(plugin_module_name)
|
|
|
|
|
log = log.replace("\\", "/")
|
2024-03-21 12:10:24 +08:00
|
|
|
|
if r:
|
2024-03-21 14:52:02 +08:00
|
|
|
|
nonebot.load_plugin(plugin_module_name) # 加载插件
|
|
|
|
|
installed_plugin = InstalledPlugin(module_name=plugin_module_name) # 构造插件信息模型
|
|
|
|
|
store_plugin = await get_store_plugin(plugin_module_name) # 获取商店中的插件信息
|
|
|
|
|
found_in_db_plugin = plugin_db.first(InstalledPlugin, "module_name = ?", plugin_module_name) # 查询数据库中是否已经安装
|
|
|
|
|
if found_in_db_plugin is None:
|
|
|
|
|
plugin_db.save(installed_plugin)
|
|
|
|
|
info = ulang.get('npm.install_success', NAME=store_plugin.name).replace('_', r'\\_') # markdown转义
|
|
|
|
|
await send_markdown(
|
|
|
|
|
f"{info}\n\n"
|
|
|
|
|
f"```\n{log}\n```",
|
|
|
|
|
bot,
|
|
|
|
|
event=event
|
|
|
|
|
)
|
|
|
|
|
else:
|
|
|
|
|
await npm_alc.finish(ulang.get('npm.plugin_already_installed', NAME=store_plugin.name))
|
2024-03-21 12:10:24 +08:00
|
|
|
|
else:
|
2024-03-21 14:52:02 +08:00
|
|
|
|
info = ulang.get('npm.install_failed', NAME=plugin_module_name).replace('_', r'\\_')
|
2024-03-21 12:10:24 +08:00
|
|
|
|
await send_markdown(
|
2024-03-21 14:52:02 +08:00
|
|
|
|
f"{info}\n\n"
|
2024-03-21 12:10:24 +08:00
|
|
|
|
f"```\n{log}\n```",
|
|
|
|
|
bot,
|
|
|
|
|
event=event
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
elif result.subcommands.get("remove"):
|
2024-03-21 14:52:02 +08:00
|
|
|
|
plugin_module_name: str = result.subcommands["remove"].args.get("plugin_name")
|
|
|
|
|
found_installed_plugin: InstalledPlugin = plugin_db.first(InstalledPlugin, "module_name = ?", plugin_module_name)
|
|
|
|
|
if found_installed_plugin:
|
|
|
|
|
plugin_db.delete(InstalledPlugin, "module_name = ?", plugin_module_name)
|
|
|
|
|
reply = f"{ulang.get('npm.remove_success', NAME=found_installed_plugin.module_name)}"
|
|
|
|
|
await npm_alc.finish(reply)
|
|
|
|
|
else:
|
|
|
|
|
await npm_alc.finish(ulang.get("npm.plugin_not_installed", NAME=plugin_module_name))
|
2024-03-21 12:10:24 +08:00
|
|
|
|
|
2024-03-21 01:20:18 +08:00
|
|
|
|
|
|
|
|
|
async def npm_update() -> bool:
|
|
|
|
|
"""
|
|
|
|
|
更新本地插件json缓存
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
bool: 是否成功更新
|
|
|
|
|
"""
|
|
|
|
|
url_list = [
|
|
|
|
|
"https://registry.nonebot.dev/plugins.json",
|
|
|
|
|
]
|
|
|
|
|
for url in url_list:
|
|
|
|
|
async with aiohttp.ClientSession() as session:
|
|
|
|
|
async with session.get(url) as resp:
|
|
|
|
|
if resp.status == 200:
|
|
|
|
|
async with aiofiles.open("data/liteyuki/plugins.json", "wb") as f:
|
|
|
|
|
data = await resp.read()
|
|
|
|
|
await f.write(data)
|
|
|
|
|
return True
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
async def npm_search(keywords: list[str]) -> list[StorePlugin]:
|
|
|
|
|
"""
|
|
|
|
|
搜索插件
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
keywords (list[str]): 关键词列表
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
list[StorePlugin]: 插件列表
|
|
|
|
|
"""
|
|
|
|
|
results = []
|
|
|
|
|
async with aiofiles.open("data/liteyuki/plugins.json", "r", encoding="utf-8") as f:
|
|
|
|
|
plugins: list[StorePlugin] = [StorePlugin(**pobj) for pobj in json.loads(await f.read())]
|
|
|
|
|
for plugin in plugins:
|
|
|
|
|
plugin_text = ' '.join(
|
|
|
|
|
[
|
|
|
|
|
plugin.name,
|
|
|
|
|
plugin.desc,
|
|
|
|
|
plugin.author,
|
|
|
|
|
plugin.module_name,
|
|
|
|
|
plugin.project_link,
|
|
|
|
|
plugin.homepage,
|
|
|
|
|
' '.join([tag.label for tag in plugin.tags])
|
|
|
|
|
]
|
|
|
|
|
)
|
|
|
|
|
if all([keyword in plugin_text for keyword in keywords]):
|
|
|
|
|
results.append(plugin)
|
|
|
|
|
return results
|
|
|
|
|
|
2024-03-20 22:30:52 +08:00
|
|
|
|
|
2024-03-21 12:10:24 +08:00
|
|
|
|
async def get_store_plugin(plugin_module_name: str) -> Optional[StorePlugin]:
|
|
|
|
|
"""
|
|
|
|
|
获取插件信息
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
plugin_module_name (str): 插件模块名
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
Optional[StorePlugin]: 插件信息
|
|
|
|
|
"""
|
|
|
|
|
async with aiofiles.open("data/liteyuki/plugins.json", "r", encoding="utf-8") as f:
|
|
|
|
|
plugins: list[StorePlugin] = [StorePlugin(**pobj) for pobj in json.loads(await f.read())]
|
|
|
|
|
for plugin in plugins:
|
|
|
|
|
if plugin.module_name == plugin_module_name:
|
|
|
|
|
return plugin
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def npm_install(plugin_module_name) -> tuple[bool, str]:
|
|
|
|
|
"""
|
|
|
|
|
Args:
|
|
|
|
|
plugin_module_name:
|
|
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
|
tuple[bool, str]:
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
buffer = StringIO()
|
|
|
|
|
sys.stdout = buffer
|
|
|
|
|
sys.stderr = buffer
|
|
|
|
|
|
|
|
|
|
mirrors = [
|
2024-03-21 14:52:02 +08:00
|
|
|
|
"https://pypi.mirrors.cqupt.edu.cn/simple", # 重庆邮电大学
|
|
|
|
|
"https://pypi.tuna.tsinghua.edu.cn/simple", # 清华大学
|
|
|
|
|
"https://pypi.liteyuki.icu/simple", # 轻雪镜像
|
|
|
|
|
"https://pypi.org/simple", # 官方源
|
2024-03-21 12:10:24 +08:00
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
# 使用pip安装包,对每个镜像尝试一次,成功后返回值
|
|
|
|
|
success = False
|
|
|
|
|
for mirror in mirrors:
|
|
|
|
|
try:
|
2024-03-21 14:52:02 +08:00
|
|
|
|
nonebot.logger.info(f"npm_install try mirror: {mirror}")
|
2024-03-21 12:10:24 +08:00
|
|
|
|
result = pip.main(['install', plugin_module_name, "-i", mirror])
|
|
|
|
|
success = result == 0
|
2024-03-21 14:52:02 +08:00
|
|
|
|
if success:
|
|
|
|
|
break
|
|
|
|
|
else:
|
|
|
|
|
nonebot.logger.warning(f"npm_install failed, try next mirror.")
|
2024-03-21 12:10:24 +08:00
|
|
|
|
except Exception as e:
|
2024-03-21 14:52:02 +08:00
|
|
|
|
|
2024-03-21 12:10:24 +08:00
|
|
|
|
success = False
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
sys.stdout = sys.__stdout__
|
|
|
|
|
sys.stderr = sys.__stderr__
|
|
|
|
|
|
|
|
|
|
return success, buffer.getvalue()
|