mirror of
https://github.com/TriM-Organization/LiteyukiBot-TriM.git
synced 2024-11-14 11:27:37 +08:00
fix: npm对插件无法启用的bug
feat: 资源包的管理器
This commit is contained in:
parent
72742d805c
commit
9743868cce
@ -7,7 +7,7 @@ category: 使用手册
|
|||||||
|
|
||||||
## 基础插件命令
|
## 基础插件命令
|
||||||
|
|
||||||
#### 命令前有[S]的表示仅超级用户可用,[O]和[A]分别为群主和群管可用
|
#### 命令前有[S]的表示仅超级用户可用,[O]和[A]分别为群主和群管可用,[P]为私聊可用
|
||||||
|
|
||||||
### 轻雪`liteyuki`
|
### 轻雪`liteyuki`
|
||||||
|
|
||||||
@ -19,31 +19,43 @@ category: 使用手册
|
|||||||
[S]config get [key] # 查询配置项,不带key返回配置项列表,推荐私聊使用
|
[S]config get [key] # 查询配置项,不带key返回配置项列表,推荐私聊使用
|
||||||
[S]reload-resources # 重载资源
|
[S]reload-resources # 重载资源
|
||||||
[S]switch-image-mode # 切换图片模式,该功能需要commit:505468b及以后的Lagrange.OneBot,在普通图片和Markdown图片之间切换,后者更大但有失败的可能
|
[S]switch-image-mode # 切换图片模式,该功能需要commit:505468b及以后的Lagrange.OneBot,在普通图片和Markdown图片之间切换,后者更大但有失败的可能
|
||||||
|
liteyuki-docs # 查看轻雪文档
|
||||||
# 上述两个命令修改的配置项在数据库中保存,但是优先级低于配置文件,如果配置文件中存在相同的配置项,将会使用配置文件中的配置
|
# 上述两个命令修改的配置项在数据库中保存,但是优先级低于配置文件,如果配置文件中存在相同的配置项,将会使用配置文件中的配置
|
||||||
------
|
------
|
||||||
别名: reload-liteyuki 重启轻雪, update-liteyuki 更新轻雪, reload-resources 重载资源, config 配置, set 设置, get 查询
|
别名: reload-liteyuki 重启轻雪, update-liteyuki 更新轻雪, reload-resources 重载资源, config 配置, set 设置, get 查询,
|
||||||
|
switch-image-mode 切换图片模式, liteyuki-docs 轻雪文档
|
||||||
```
|
```
|
||||||
|
|
||||||
### 轻雪Nonebot插件管理 `liteyuki_npm`
|
### 轻雪包管理器 `liteyuki_npm`
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
[S]npm update # 更新插件索引
|
[S]nps update # 更新插件索引
|
||||||
[S]npm install <plugin_name> # 安装插件
|
[S]nps install <plugin_name> # 安装插件
|
||||||
[S]npm uninstall <plugin_name> # 卸载插件
|
[S]nps uninstall <plugin_name> # 卸载插件
|
||||||
[S]npm search <keywords...> # 通过关键词搜索插件
|
[S]nps search <keywords...> # 通过关键词搜索插件
|
||||||
------
|
------
|
||||||
别名: npm 插件, update 更新, install 安装, uninstall 卸载, search 搜索
|
[AOSP]npm enable <plugin_name> # 当前会话启用插件
|
||||||
|
[AOSP]npm disable <plugin_name> # 当前会话禁用插件
|
||||||
|
[S]npm enable-global <plugin_name> # 全局启用插件
|
||||||
|
[S]npm disable-global <plugin_name> # 全局禁用插件
|
||||||
|
list-plugin [page] [num] # 列出所有插件 page为页数,num为每页显示数量
|
||||||
|
------
|
||||||
|
[S]rpm list [page] [num] # 列出所有资源包 page为页数,num为每页显示数量
|
||||||
|
[S]rpm load <resource_pack_name> # 加载资源包
|
||||||
|
[S]rpm unload <resource_pack_name> # 卸载资源包
|
||||||
|
[S]rpm change <resource_pack_name> # 修改优先级
|
||||||
|
[S]rpm reload # 重载所有资源包
|
||||||
|
------
|
||||||
|
别名: nps 插件商店, npm 插件管理, update 更新, install 安装, uninstall 卸载, search 搜索,
|
||||||
|
enable 启用, disable 停用, enable-global 全局启用, disable-global 全局停用, list-plugin 列出插件/插件列表,
|
||||||
|
rpm 资源包, load 加载, unload 卸载, change 更改, reload 重载, list 列表/列出
|
||||||
```
|
```
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
[SOA]enable <plugin_name> # 启用插件
|
|
||||||
[SOA]disable <plugin_name> # 禁用插件
|
|
||||||
[S]enable-global <plugin_name> # 全局启用插件
|
|
||||||
[S]disable-global <plugin_name> # 全局禁用插件
|
|
||||||
list-plugin [page] [num] # 列出所有插件 page为页数,num为每页显示数量
|
|
||||||
# 受限于Nonebot的钩子函数,目前只能阻断消息事件的传入,对于主动推送消息的插件,无法将其阻止
|
# 受限于Nonebot的钩子函数,目前只能阻断消息事件的传入,对于主动推送消息的插件,无法将其阻止
|
||||||
------
|
------
|
||||||
别名: enable 启用, disable 停用, enable-global 全局启用, disable-global 全局停用, list-plugin 列出插件/插件列表
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### 轻雪用户管理`liteyuki_user`
|
### 轻雪用户管理`liteyuki_user`
|
||||||
|
@ -6,6 +6,7 @@ import pip
|
|||||||
from git import Repo
|
from git import Repo
|
||||||
from nonebot import Bot, require, get_driver
|
from nonebot import Bot, require, get_driver
|
||||||
from nonebot.exception import MockApiException
|
from nonebot.exception import MockApiException
|
||||||
|
from nonebot.internal.matcher import Matcher
|
||||||
from nonebot.permission import SUPERUSER
|
from nonebot.permission import SUPERUSER
|
||||||
|
|
||||||
from liteyuki.utils.config import config, load_from_yaml
|
from liteyuki.utils.config import config, load_from_yaml
|
||||||
@ -23,71 +24,23 @@ driver = get_driver()
|
|||||||
|
|
||||||
markdown_image = common_db.first(StoredConfig(), default=StoredConfig()).config.get("markdown_image", False)
|
markdown_image = common_db.first(StoredConfig(), default=StoredConfig()).config.get("markdown_image", False)
|
||||||
|
|
||||||
liteyuki = on_alconna(
|
@on_alconna(
|
||||||
command=Alconna(
|
command=Alconna(
|
||||||
"liteecho",
|
"liteecho",
|
||||||
),
|
),
|
||||||
permission=SUPERUSER
|
permission=SUPERUSER
|
||||||
)
|
).handle()
|
||||||
|
async def _(bot: T_Bot, matcher: Matcher):
|
||||||
|
await matcher.finish(f"Hello, Liteyuki!\nBot {bot.self_id}")
|
||||||
|
|
||||||
update_liteyuki = on_alconna(
|
|
||||||
|
@on_alconna(
|
||||||
aliases={"更新轻雪"},
|
aliases={"更新轻雪"},
|
||||||
command=Alconna(
|
command=Alconna(
|
||||||
"update-liteyuki"
|
"update-liteyuki"
|
||||||
),
|
),
|
||||||
permission=SUPERUSER
|
permission=SUPERUSER
|
||||||
)
|
).handle()
|
||||||
|
|
||||||
reload_liteyuki = on_alconna(
|
|
||||||
aliases={"重启轻雪"},
|
|
||||||
command=Alconna(
|
|
||||||
"reload-liteyuki"
|
|
||||||
),
|
|
||||||
permission=SUPERUSER
|
|
||||||
)
|
|
||||||
|
|
||||||
reload_resources = on_alconna(
|
|
||||||
aliases={"重载资源"},
|
|
||||||
command=Alconna(
|
|
||||||
"reload-resources"
|
|
||||||
),
|
|
||||||
permission=SUPERUSER
|
|
||||||
)
|
|
||||||
|
|
||||||
cmd_config = on_alconna(
|
|
||||||
aliases={"配置"},
|
|
||||||
command=Alconna(
|
|
||||||
"config",
|
|
||||||
Subcommand(
|
|
||||||
"set",
|
|
||||||
Args["key", str]["value", Any],
|
|
||||||
alias=["设置"],
|
|
||||||
|
|
||||||
),
|
|
||||||
Subcommand(
|
|
||||||
"get",
|
|
||||||
Args["key", str, None],
|
|
||||||
alias=["查询"]
|
|
||||||
)
|
|
||||||
),
|
|
||||||
permission=SUPERUSER
|
|
||||||
)
|
|
||||||
|
|
||||||
switch_image_mode = on_alconna(
|
|
||||||
aliases={"切换图片模式"},
|
|
||||||
command=Alconna(
|
|
||||||
"switch-image-mode"
|
|
||||||
),
|
|
||||||
permission=SUPERUSER
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@liteyuki.handle()
|
|
||||||
async def _(bot: T_Bot):
|
|
||||||
await liteyuki.finish(f"Hello, Liteyuki!\nBot {bot.self_id}")
|
|
||||||
|
|
||||||
|
|
||||||
@update_liteyuki.handle()
|
|
||||||
async def _(bot: T_Bot, event: T_MessageEvent):
|
async def _(bot: T_Bot, event: T_MessageEvent):
|
||||||
# 使用git pull更新
|
# 使用git pull更新
|
||||||
ulang = get_user_lang(str(event.user_id))
|
ulang = get_user_lang(str(event.user_id))
|
||||||
@ -117,20 +70,43 @@ async def _(bot: T_Bot, event: T_MessageEvent):
|
|||||||
nonebot.logger.error(f"Pull from {origin} failed: {e}")
|
nonebot.logger.error(f"Pull from {origin} failed: {e}")
|
||||||
reply = "Liteyuki updated!\n"
|
reply = "Liteyuki updated!\n"
|
||||||
reply += f"```\n{logs}\n```\n"
|
reply += f"```\n{logs}\n```\n"
|
||||||
btn_restart = md.cmd(ulang.get("liteyuki.restart_now"), "reload-liteyuki")
|
btn_restart = md.btn_cmd(ulang.get("liteyuki.restart_now"), "reload-liteyuki")
|
||||||
pip.main(["install", "-r", "requirements.txt"])
|
pip.main(["install", "-r", "requirements.txt"])
|
||||||
reply += f"{ulang.get('liteyuki.update_restart', RESTART=btn_restart)}"
|
reply += f"{ulang.get('liteyuki.update_restart', RESTART=btn_restart)}"
|
||||||
await md.send_md(reply, bot, event=event, at_sender=False)
|
await md.send_md(reply, bot, event=event, at_sender=False)
|
||||||
|
|
||||||
|
|
||||||
@reload_liteyuki.handle()
|
@on_alconna(
|
||||||
async def _():
|
aliases={"重启轻雪"},
|
||||||
await reload_liteyuki.send("Liteyuki reloading")
|
command=Alconna(
|
||||||
|
"reload-liteyuki"
|
||||||
|
),
|
||||||
|
permission=SUPERUSER
|
||||||
|
).handle()
|
||||||
|
async def _(matcher: Matcher):
|
||||||
|
await matcher.send("Liteyuki reloading")
|
||||||
Reloader.reload(3)
|
Reloader.reload(3)
|
||||||
|
|
||||||
|
|
||||||
@cmd_config.handle()
|
@on_alconna(
|
||||||
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
aliases={"配置"},
|
||||||
|
command=Alconna(
|
||||||
|
"config",
|
||||||
|
Subcommand(
|
||||||
|
"set",
|
||||||
|
Args["key", str]["value", Any],
|
||||||
|
alias=["设置"],
|
||||||
|
|
||||||
|
),
|
||||||
|
Subcommand(
|
||||||
|
"get",
|
||||||
|
Args["key", str, None],
|
||||||
|
alias=["查询"]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
permission=SUPERUSER
|
||||||
|
).handle()
|
||||||
|
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot, matcher: Matcher):
|
||||||
ulang = get_user_lang(str(event.user_id))
|
ulang = get_user_lang(str(event.user_id))
|
||||||
stored_config: StoredConfig = common_db.first(StoredConfig(), default=StoredConfig())
|
stored_config: StoredConfig = common_db.first(StoredConfig(), default=StoredConfig())
|
||||||
if result.subcommands.get("set"):
|
if result.subcommands.get("set"):
|
||||||
@ -141,7 +117,7 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
pass
|
pass
|
||||||
stored_config.config[key] = value
|
stored_config.config[key] = value
|
||||||
common_db.upsert(stored_config)
|
common_db.upsert(stored_config)
|
||||||
await cmd_config.finish(f"{ulang.get('liteyuki.config_set_success', KEY=key, VAL=value)}")
|
await matcher.finish(f"{ulang.get('liteyuki.config_set_success', KEY=key, VAL=value)}")
|
||||||
elif result.subcommands.get("get"):
|
elif result.subcommands.get("get"):
|
||||||
key = result.subcommands.get("get").args.get("key")
|
key = result.subcommands.get("get").args.get("key")
|
||||||
file_config = load_from_yaml("config.yml")
|
file_config = load_from_yaml("config.yml")
|
||||||
@ -162,19 +138,14 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
await md.send_md(reply, bot, event=event)
|
await md.send_md(reply, bot, event=event)
|
||||||
|
|
||||||
|
|
||||||
@reload_resources.handle()
|
@on_alconna(
|
||||||
async def _(event: T_MessageEvent):
|
aliases={"切换图片模式"},
|
||||||
ulang = get_user_lang(str(event.user_id))
|
command=Alconna(
|
||||||
load_resources()
|
"switch-image-mode"
|
||||||
await reload_resources.finish(
|
),
|
||||||
ulang.get("liteyuki.reload_resources_success",
|
permission=SUPERUSER
|
||||||
NUM=len(get_loaded_resource_packs())
|
).handle()
|
||||||
)
|
async def _(event: T_MessageEvent, matcher: Matcher):
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@switch_image_mode.handle()
|
|
||||||
async def _(bot: T_Bot, event: T_MessageEvent):
|
|
||||||
global markdown_image
|
global markdown_image
|
||||||
# 切换图片模式,False以图片形式发送,True以markdown形式发送
|
# 切换图片模式,False以图片形式发送,True以markdown形式发送
|
||||||
ulang = get_user_lang(str(event.user_id))
|
ulang = get_user_lang(str(event.user_id))
|
||||||
@ -182,12 +153,20 @@ async def _(bot: T_Bot, event: T_MessageEvent):
|
|||||||
stored_config.config["markdown_image"] = not stored_config.config.get("markdown_image", False)
|
stored_config.config["markdown_image"] = not stored_config.config.get("markdown_image", False)
|
||||||
markdown_image = stored_config.config["markdown_image"]
|
markdown_image = stored_config.config["markdown_image"]
|
||||||
common_db.upsert(stored_config)
|
common_db.upsert(stored_config)
|
||||||
await switch_image_mode.finish(ulang.get("liteyuki.image_mode_on" if stored_config.config["markdown_image"] else "liteyuki.image_mode_off"))
|
await matcher.finish(ulang.get("liteyuki.image_mode_on" if stored_config.config["markdown_image"] else "liteyuki.image_mode_off"))
|
||||||
|
|
||||||
|
|
||||||
|
@on_alconna(
|
||||||
|
command=Alconna(
|
||||||
|
"liteyuki-docs",
|
||||||
|
),
|
||||||
|
aliases={"轻雪文档"},
|
||||||
|
).handle()
|
||||||
|
async def _(matcher: Matcher):
|
||||||
|
matcher.finish("https://bot.liteyuki.icu/usage")
|
||||||
|
|
||||||
# system hook
|
# system hook
|
||||||
|
@Bot.on_calling_api # 图片模式检测
|
||||||
@Bot.on_calling_api
|
|
||||||
async def test_for_md_image(bot: T_Bot, api: str, data: dict):
|
async def test_for_md_image(bot: T_Bot, api: str, data: dict):
|
||||||
if api in ["send_msg", "send_private_msg", "send_group_msg"] and markdown_image and data.get("user_id") != bot.self_id:
|
if api in ["send_msg", "send_private_msg", "send_group_msg"] and markdown_image and data.get("user_id") != bot.self_id:
|
||||||
if api == "send_msg" and data.get("message_type") == "private" or api == "send_private_msg":
|
if api == "send_msg" and data.get("message_type") == "private" or api == "send_private_msg":
|
||||||
|
@ -37,7 +37,8 @@ async def _(bot: T_Bot, event: T_MessageEvent):
|
|||||||
{
|
{
|
||||||
"data": await get_stats_data(bot.self_id, ulang.lang_code)
|
"data": await get_stats_data(bot.self_id, ulang.lang_code)
|
||||||
},
|
},
|
||||||
debug=True
|
debug=True,
|
||||||
|
wait=1
|
||||||
)
|
)
|
||||||
await stats.finish(MessageSegment.image(image))
|
await stats.finish(MessageSegment.image(image))
|
||||||
|
|
||||||
@ -233,10 +234,4 @@ async def get_stats_data(self_id: str = None, lang: str = None) -> dict:
|
|||||||
"total_trans": ulang.get("main.monitor.total"),
|
"total_trans": ulang.get("main.monitor.total"),
|
||||||
}
|
}
|
||||||
|
|
||||||
# for ps_name, ps_mem in process_mem.items():
|
|
||||||
# templ["memTags"].insert(
|
|
||||||
# 0,
|
|
||||||
# f"{ps_name} {convert_size(ps_mem, 1)}"
|
|
||||||
# )
|
|
||||||
|
|
||||||
return templ
|
return templ
|
||||||
|
@ -157,14 +157,14 @@ class Minesweeper:
|
|||||||
print([d.value for d in row])
|
print([d.value for d in row])
|
||||||
for dot in row:
|
for dot in row:
|
||||||
if dot.mask and not dot.flagged:
|
if dot.mask and not dot.flagged:
|
||||||
text += md.cmd(self.MASK, f"minesweeper reveal {dot.row} {dot.col}")
|
text += md.btn_cmd(self.MASK, f"minesweeper reveal {dot.row} {dot.col}")
|
||||||
elif dot.flagged:
|
elif dot.flagged:
|
||||||
text += md.cmd(self.FLAG, f"minesweeper mark {dot.row} {dot.col}")
|
text += md.btn_cmd(self.FLAG, f"minesweeper mark {dot.row} {dot.col}")
|
||||||
else:
|
else:
|
||||||
text += self.NUMS[dot.value]
|
text += self.NUMS[dot.value]
|
||||||
text += dis
|
text += dis
|
||||||
text += "\n"
|
text += "\n"
|
||||||
btn_mark = md.cmd("标记", f"minesweeper mark ", enter=False)
|
btn_mark = md.btn_cmd("标记", f"minesweeper mark ", enter=False)
|
||||||
btn_end = md.cmd("结束", "minesweeper end", enter=True)
|
btn_end = md.btn_cmd("结束", "minesweeper end", enter=True)
|
||||||
text += f" {btn_mark} {btn_end}"
|
text += f" {btn_mark} {btn_end}"
|
||||||
return text
|
return text
|
||||||
|
@ -16,9 +16,9 @@ from .common import *
|
|||||||
require("nonebot_plugin_alconna")
|
require("nonebot_plugin_alconna")
|
||||||
from nonebot_plugin_alconna import Alconna, Args, Subcommand, on_alconna
|
from nonebot_plugin_alconna import Alconna, Args, Subcommand, on_alconna
|
||||||
|
|
||||||
npm_alc = on_alconna(
|
nps = on_alconna(
|
||||||
Alconna(
|
Alconna(
|
||||||
"npm",
|
"nps",
|
||||||
Subcommand(
|
Subcommand(
|
||||||
"update",
|
"update",
|
||||||
alias=["u"],
|
alias=["u"],
|
||||||
@ -36,19 +36,15 @@ npm_alc = on_alconna(
|
|||||||
Subcommand(
|
Subcommand(
|
||||||
"uninstall",
|
"uninstall",
|
||||||
Args["plugin_name", str],
|
Args["plugin_name", str],
|
||||||
alias=["rm", "移除", "卸载"],
|
alias=["r", "rm", "卸载"],
|
||||||
),
|
|
||||||
Subcommand(
|
|
||||||
"list",
|
|
||||||
alias=["l", "ls", "列表"],
|
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
aliases={"插件"},
|
aliases={"插件商店"},
|
||||||
permission=SUPERUSER,
|
permission=SUPERUSER,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@npm_alc.handle()
|
@nps.handle()
|
||||||
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
||||||
ulang = get_user_lang(str(event.user_id))
|
ulang = get_user_lang(str(event.user_id))
|
||||||
|
|
||||||
@ -58,9 +54,9 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
if result.subcommands.get("update"):
|
if result.subcommands.get("update"):
|
||||||
r = await npm_update()
|
r = await npm_update()
|
||||||
if r:
|
if r:
|
||||||
await npm_alc.finish(ulang.get("npm.store_update_success"))
|
await nps.finish(ulang.get("npm.store_update_success"))
|
||||||
else:
|
else:
|
||||||
await npm_alc.finish(ulang.get("npm.store_update_failed"))
|
await nps.finish(ulang.get("npm.store_update_failed"))
|
||||||
|
|
||||||
elif result.subcommands.get("search"):
|
elif result.subcommands.get("search"):
|
||||||
keywords: list[str] = result.subcommands["search"].args.get("keywords")
|
keywords: list[str] = result.subcommands["search"].args.get("keywords")
|
||||||
@ -71,9 +67,9 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
if len(rs):
|
if len(rs):
|
||||||
reply = f"{ulang.get('npm.search_result')} | {ulang.get('npm.total', TOTAL=len(rs))}\n***"
|
reply = f"{ulang.get('npm.search_result')} | {ulang.get('npm.total', TOTAL=len(rs))}\n***"
|
||||||
for plugin in rs[:min(max_show, len(rs))]:
|
for plugin in rs[:min(max_show, len(rs))]:
|
||||||
btn_install = md.cmd(ulang.get("npm.install"), "npm install %s" % plugin.module_name)
|
btn_install = md.btn_cmd(ulang.get("npm.install"), "npm install %s" % plugin.module_name)
|
||||||
link_page = md.link(ulang.get("npm.homepage"), plugin.homepage)
|
link_page = md.btn_link(ulang.get("npm.homepage"), plugin.homepage)
|
||||||
link_pypi = md.link(ulang.get("npm.pypi"), plugin.homepage)
|
link_pypi = md.btn_link(ulang.get("npm.pypi"), plugin.homepage)
|
||||||
|
|
||||||
reply += (f"\n# **{plugin.name}**\n"
|
reply += (f"\n# **{plugin.name}**\n"
|
||||||
f"\n> **{plugin.desc}**\n"
|
f"\n> **{plugin.desc}**\n"
|
||||||
@ -89,14 +85,14 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
elif result.subcommands.get("install"):
|
elif result.subcommands.get("install"):
|
||||||
plugin_module_name: str = result.subcommands["install"].args.get("plugin_name")
|
plugin_module_name: str = result.subcommands["install"].args.get("plugin_name")
|
||||||
store_plugin = await get_store_plugin(plugin_module_name)
|
store_plugin = await get_store_plugin(plugin_module_name)
|
||||||
await npm_alc.send(ulang.get("npm.installing", NAME=plugin_module_name))
|
await nps.send(ulang.get("npm.installing", NAME=plugin_module_name))
|
||||||
r, log = npm_install(plugin_module_name)
|
r, log = npm_install(plugin_module_name)
|
||||||
log = log.replace("\\", "/")
|
log = log.replace("\\", "/")
|
||||||
|
|
||||||
if not store_plugin:
|
if not store_plugin:
|
||||||
await npm_alc.finish(ulang.get("npm.plugin_not_found", NAME=plugin_module_name))
|
await nps.finish(ulang.get("npm.plugin_not_found", NAME=plugin_module_name))
|
||||||
|
|
||||||
homepage_btn = md.cmd(ulang.get("npm.homepage"), store_plugin.homepage)
|
homepage_btn = md.btn_cmd(ulang.get("npm.homepage"), store_plugin.homepage)
|
||||||
if r:
|
if r:
|
||||||
|
|
||||||
r_load = nonebot.load_plugin(plugin_module_name) # 加载插件
|
r_load = nonebot.load_plugin(plugin_module_name) # 加载插件
|
||||||
@ -114,7 +110,7 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
event=event
|
event=event
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
await npm_alc.finish(ulang.get("npm.plugin_already_installed", NAME=store_plugin.name))
|
await nps.finish(ulang.get("npm.plugin_already_installed", NAME=store_plugin.name))
|
||||||
else:
|
else:
|
||||||
info = ulang.get("npm.load_failed", NAME=plugin_module_name, HOMEPAGE=homepage_btn).replace("_", r"\\_")
|
info = ulang.get("npm.load_failed", NAME=plugin_module_name, HOMEPAGE=homepage_btn).replace("_", r"\\_")
|
||||||
await md.send_md(
|
await md.send_md(
|
||||||
@ -138,9 +134,9 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
if found_installed_plugin:
|
if found_installed_plugin:
|
||||||
plugin_db.delete(InstalledPlugin(), "module_name = ?", plugin_module_name)
|
plugin_db.delete(InstalledPlugin(), "module_name = ?", plugin_module_name)
|
||||||
reply = f"{ulang.get('npm.uninstall_success', NAME=found_installed_plugin.module_name)}"
|
reply = f"{ulang.get('npm.uninstall_success', NAME=found_installed_plugin.module_name)}"
|
||||||
await npm_alc.finish(reply)
|
await nps.finish(reply)
|
||||||
else:
|
else:
|
||||||
await npm_alc.finish(ulang.get("npm.plugin_not_installed", NAME=plugin_module_name))
|
await nps.finish(ulang.get("npm.plugin_not_installed", NAME=plugin_module_name))
|
||||||
|
|
||||||
|
|
||||||
async def npm_update() -> bool:
|
async def npm_update() -> bool:
|
||||||
|
@ -19,7 +19,7 @@ from .installer import get_store_plugin, npm_update
|
|||||||
from ...utils.tools import clamp
|
from ...utils.tools import clamp
|
||||||
|
|
||||||
require("nonebot_plugin_alconna")
|
require("nonebot_plugin_alconna")
|
||||||
from nonebot_plugin_alconna import on_alconna, Alconna, Args, Arparma
|
from nonebot_plugin_alconna import on_alconna, Alconna, Args, Arparma, Subcommand
|
||||||
|
|
||||||
list_plugins = on_alconna(
|
list_plugins = on_alconna(
|
||||||
Alconna(
|
Alconna(
|
||||||
@ -29,27 +29,40 @@ list_plugins = on_alconna(
|
|||||||
aliases={"列出插件", "插件列表"}
|
aliases={"列出插件", "插件列表"}
|
||||||
)
|
)
|
||||||
|
|
||||||
toggle_plugin = on_alconna(
|
npm = on_alconna(
|
||||||
Alconna(
|
aliases={"插件管理"},
|
||||||
|
command=Alconna(
|
||||||
|
"npm",
|
||||||
|
# Args["plugin_name", str],
|
||||||
|
Subcommand(
|
||||||
"enable",
|
"enable",
|
||||||
Args["plugin_name", str],
|
Args["plugin_name", str],
|
||||||
),
|
alias=["启用"],
|
||||||
aliases={"disable", "启用", "停用"}
|
|
||||||
)
|
|
||||||
|
|
||||||
toggle_plugin_global = on_alconna(
|
|
||||||
Alconna(
|
|
||||||
"enable-global",
|
|
||||||
Args["plugin_name", str],
|
|
||||||
),
|
),
|
||||||
permission=SUPERUSER,
|
Subcommand(
|
||||||
aliases={"disable-global", "全局启用", "全局停用"}
|
"disable",
|
||||||
|
Args["plugin_name", str],
|
||||||
|
alias=["停用"],
|
||||||
|
),
|
||||||
|
Subcommand(
|
||||||
|
"global-enable",
|
||||||
|
Args["plugin_name", str],
|
||||||
|
alias=["全局启用"],
|
||||||
|
),
|
||||||
|
Subcommand(
|
||||||
|
"global-disable",
|
||||||
|
Args["plugin_name", str],
|
||||||
|
alias=["全局停用"],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@list_plugins.handle()
|
@list_plugins.handle()
|
||||||
async def _(event: T_MessageEvent, bot: T_Bot, result: Arparma):
|
async def _(event: T_MessageEvent, bot: T_Bot, result: Arparma):
|
||||||
lang = get_user_lang(str(event.user_id))
|
ulang = get_user_lang(str(event.user_id))
|
||||||
if not os.path.exists("data/liteyuki/plugins.json"):
|
if not os.path.exists("data/liteyuki/plugins.json"):
|
||||||
await npm_update()
|
await npm_update()
|
||||||
|
|
||||||
@ -60,31 +73,29 @@ async def _(event: T_MessageEvent, bot: T_Bot, result: Arparma):
|
|||||||
page = clamp(result.args.get("page"), 1, total)
|
page = clamp(result.args.get("page"), 1, total)
|
||||||
|
|
||||||
# 已加载插件 | 总计10 | 第1/3页
|
# 已加载插件 | 总计10 | 第1/3页
|
||||||
reply = (f"# {lang.get('npm.loaded_plugins')} | "
|
reply = (f"# {ulang.get('npm.loaded_plugins')} | "
|
||||||
f"{lang.get('npm.total', TOTAL=len(nonebot.get_loaded_plugins()))} | "
|
f"{ulang.get('npm.total', TOTAL=len(nonebot.get_loaded_plugins()))} | "
|
||||||
f"{lang.get('npm.page', PAGE=page, TOTAL=total)} \n***\n")
|
f"{ulang.get('npm.page', PAGE=page, TOTAL=total)} \n***\n")
|
||||||
|
|
||||||
for plugin in loaded_plugin_list[(page - 1) * num_per_page: min(page * num_per_page, len(loaded_plugin_list))]:
|
for plugin in loaded_plugin_list[(page - 1) * num_per_page: min(page * num_per_page, len(loaded_plugin_list))]:
|
||||||
# 检查是否有 metadata 属性
|
# 检查是否有 metadata 属性
|
||||||
# 添加帮助按钮
|
# 添加帮助按钮
|
||||||
btn_usage = md.cmd(lang.get("npm.usage"), f"help {plugin.module_name}", False)
|
btn_usage = md.btn_cmd(ulang.get("npm.usage"), f"help {plugin.module_name}", False)
|
||||||
store_plugin = await get_store_plugin(plugin.module_name)
|
store_plugin = await get_store_plugin(plugin.module_name)
|
||||||
|
|
||||||
session_enable = get_plugin_session_enable(event, plugin.module_name)
|
session_enable = get_plugin_session_enable(event, plugin.module_name)
|
||||||
|
|
||||||
if store_plugin:
|
if store_plugin:
|
||||||
btn_homepage = md.link(lang.get("npm.homepage"), store_plugin.homepage)
|
btn_homepage = md.btn_link(ulang.get("npm.homepage"), store_plugin.homepage)
|
||||||
show_name = store_plugin.name
|
show_name = store_plugin.name
|
||||||
elif plugin.metadata:
|
elif plugin.metadata:
|
||||||
if plugin.metadata.extra.get("liteyuki"):
|
if plugin.metadata.extra.get("liteyuki"):
|
||||||
btn_homepage = md.link(lang.get("npm.homepage"), "https://github.com/snowykami/LiteyukiBot")
|
btn_homepage = md.btn_link(ulang.get("npm.homepage"), "https://github.com/snowykami/LiteyukiBot")
|
||||||
else:
|
else:
|
||||||
btn_homepage = lang.get("npm.homepage")
|
btn_homepage = ulang.get("npm.homepage")
|
||||||
show_name = plugin.metadata.name
|
show_name = plugin.metadata.name
|
||||||
else:
|
else:
|
||||||
btn_homepage = lang.get("npm.homepage")
|
btn_homepage = ulang.get("npm.homepage")
|
||||||
show_name = plugin.name
|
show_name = plugin.name
|
||||||
lang.get("npm.no_description")
|
ulang.get("npm.no_description")
|
||||||
|
|
||||||
if plugin.metadata:
|
if plugin.metadata:
|
||||||
reply += f"\n**{md.escape(show_name)}**\n"
|
reply += f"\n**{md.escape(show_name)}**\n"
|
||||||
@ -95,11 +106,10 @@ async def _(event: T_MessageEvent, bot: T_Bot, result: Arparma):
|
|||||||
|
|
||||||
if await GROUP_ADMIN(bot, event) or await GROUP_OWNER(bot, event) or await SUPERUSER(bot, event):
|
if await GROUP_ADMIN(bot, event) or await GROUP_OWNER(bot, event) or await SUPERUSER(bot, event):
|
||||||
# 添加启用/停用插件按钮
|
# 添加启用/停用插件按钮
|
||||||
cmd_toggle = f"{'disable' if session_enable else 'enable'} {plugin.module_name}"
|
cmd_toggle = f"npm {'disable' if session_enable else 'enable'} {plugin.module_name}"
|
||||||
text_toggle = lang.get("npm.disable" if session_enable else "npm.enable")
|
text_toggle = ulang.get("npm.disable" if session_enable else "npm.enable")
|
||||||
can_be_toggle = get_plugin_can_be_toggle(plugin.module_name)
|
can_be_toggle = get_plugin_can_be_toggle(plugin.module_name)
|
||||||
btn_toggle = text_toggle if not can_be_toggle else md.cmd(text_toggle, cmd_toggle)
|
btn_toggle = text_toggle if not can_be_toggle else md.btn_cmd(text_toggle, cmd_toggle)
|
||||||
|
|
||||||
reply += f" {btn_toggle}"
|
reply += f" {btn_toggle}"
|
||||||
|
|
||||||
if await SUPERUSER(bot, event):
|
if await SUPERUSER(bot, event):
|
||||||
@ -107,20 +117,18 @@ async def _(event: T_MessageEvent, bot: T_Bot, result: Arparma):
|
|||||||
# 添加移除插件和全局切换按钮
|
# 添加移除插件和全局切换按钮
|
||||||
global_enable = get_plugin_global_enable(plugin.module_name)
|
global_enable = get_plugin_global_enable(plugin.module_name)
|
||||||
btn_uninstall = (
|
btn_uninstall = (
|
||||||
md.cmd(lang.get("npm.uninstall"), f'npm uninstall {plugin.module_name}')) if plugin_in_database else lang.get(
|
md.btn_cmd(ulang.get("npm.uninstall"), f'npm uninstall {plugin.module_name}')) if plugin_in_database else ulang.get(
|
||||||
'npm.uninstall')
|
'npm.uninstall')
|
||||||
|
btn_toggle_global_text = ulang.get("npm.disable_global" if global_enable else "npm.enable_global")
|
||||||
btn_toggle_global_text = lang.get("npm.disable_global" if global_enable else "npm.enable_global")
|
cmd_toggle_global = f"npm {'global-disable' if global_enable else 'global-enable'} {plugin.module_name}"
|
||||||
cmd_toggle_global = f"{'disable-global' if global_enable else 'enable-global'} {plugin.module_name}"
|
btn_toggle_global = btn_toggle_global_text if not can_be_toggle else md.btn_cmd(btn_toggle_global_text, cmd_toggle_global)
|
||||||
btn_toggle_global = btn_toggle_global_text if not can_be_toggle else md.cmd(btn_toggle_global_text, cmd_toggle_global)
|
|
||||||
|
|
||||||
reply += f" {btn_uninstall} {btn_toggle_global}"
|
reply += f" {btn_uninstall} {btn_toggle_global}"
|
||||||
|
|
||||||
reply += "\n\n***\n"
|
reply += "\n\n***\n"
|
||||||
await md.send_md(reply, bot, event=event)
|
await md.send_md(reply, bot, event=event)
|
||||||
|
|
||||||
|
|
||||||
@toggle_plugin.handle()
|
@npm.handle()
|
||||||
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
||||||
if not os.path.exists("data/liteyuki/plugins.json"):
|
if not os.path.exists("data/liteyuki/plugins.json"):
|
||||||
await npm_update()
|
await npm_update()
|
||||||
@ -128,7 +136,9 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
ulang = get_user_lang(str(event.user_id))
|
ulang = get_user_lang(str(event.user_id))
|
||||||
plugin_module_name = result.args.get("plugin_name")
|
plugin_module_name = result.args.get("plugin_name")
|
||||||
# 支持对自定义command_start的判断
|
# 支持对自定义command_start的判断
|
||||||
toggle = result.header_result in [prefix + header for prefix in bot.config.command_start for header in ["enable-plugin", "启用"]] # 判断是启用还是停用
|
if result.subcommands.get("enable") or result.subcommands.get("disable"):
|
||||||
|
|
||||||
|
toggle = result.subcommands.get("enable") is not None
|
||||||
|
|
||||||
session_enable = get_plugin_session_enable(event, plugin_module_name) # 获取插件当前状态
|
session_enable = get_plugin_session_enable(event, plugin_module_name) # 获取插件当前状态
|
||||||
|
|
||||||
@ -137,10 +147,10 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
can_be_toggled = get_plugin_can_be_toggle(plugin_module_name) # 获取插件是否可以被启用/停用
|
can_be_toggled = get_plugin_can_be_toggle(plugin_module_name) # 获取插件是否可以被启用/停用
|
||||||
|
|
||||||
if not can_be_toggled:
|
if not can_be_toggled:
|
||||||
await toggle_plugin.finish(ulang.get("npm.plugin_cannot_be_toggled", NAME=plugin_module_name))
|
await npm.finish(ulang.get("npm.plugin_cannot_be_toggled", NAME=plugin_module_name))
|
||||||
|
|
||||||
if session_enable == toggle:
|
if session_enable == toggle:
|
||||||
await toggle_plugin.finish(
|
await npm.finish(
|
||||||
ulang.get("npm.plugin_already", NAME=plugin_module_name, STATUS=ulang.get("npm.enable") if toggle else ulang.get("npm.disable")))
|
ulang.get("npm.plugin_already", NAME=plugin_module_name, STATUS=ulang.get("npm.enable") if toggle else ulang.get("npm.disable")))
|
||||||
|
|
||||||
if event.message_type == "private":
|
if event.message_type == "private":
|
||||||
@ -167,7 +177,7 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
group_db.upsert(session)
|
group_db.upsert(session)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print(e)
|
print(e)
|
||||||
await toggle_plugin.finish(
|
await npm.finish(
|
||||||
ulang.get(
|
ulang.get(
|
||||||
"npm.toggle_failed",
|
"npm.toggle_failed",
|
||||||
NAME=plugin_module_name,
|
NAME=plugin_module_name,
|
||||||
@ -175,30 +185,21 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
ERROR=str(e))
|
ERROR=str(e))
|
||||||
)
|
)
|
||||||
|
|
||||||
await toggle_plugin.finish(
|
await npm.finish(
|
||||||
ulang.get(
|
ulang.get(
|
||||||
"npm.toggle_success",
|
"npm.toggle_success",
|
||||||
NAME=plugin_module_name,
|
NAME=plugin_module_name,
|
||||||
STATUS=ulang.get("npm.enable") if toggle else ulang.get("npm.disable"))
|
STATUS=ulang.get("npm.enable") if toggle else ulang.get("npm.disable"))
|
||||||
)
|
)
|
||||||
|
elif result.subcommands.get("global-enable") or result.subcommands.get("global-disable") and await SUPERUSER(bot, event):
|
||||||
|
toggle = result.subcommands.get("global-enable") is not None
|
||||||
@toggle_plugin_global.handle()
|
|
||||||
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|
||||||
if not os.path.exists("data/liteyuki/plugins.json"):
|
|
||||||
await npm_update()
|
|
||||||
# 判断会话类型
|
|
||||||
ulang = get_user_lang(str(event.user_id))
|
|
||||||
plugin_module_name = result.args.get("plugin_name")
|
|
||||||
# 支持对自定义command_start的判断
|
|
||||||
toggle = result.header_result in [prefix + header for prefix in bot.config.command_start for header in ["enable-global", "全局启用"]]
|
|
||||||
can_be_toggled = get_plugin_can_be_toggle(plugin_module_name)
|
can_be_toggled = get_plugin_can_be_toggle(plugin_module_name)
|
||||||
if not can_be_toggled:
|
if not can_be_toggled:
|
||||||
await toggle_plugin_global.finish(ulang.get("npm.plugin_cannot_be_toggled", NAME=plugin_module_name))
|
await npm.finish(ulang.get("npm.plugin_cannot_be_toggled", NAME=plugin_module_name))
|
||||||
|
|
||||||
global_enable = get_plugin_global_enable(plugin_module_name)
|
global_enable = get_plugin_global_enable(plugin_module_name)
|
||||||
if global_enable == toggle:
|
if global_enable == toggle:
|
||||||
await toggle_plugin_global.finish(
|
await npm.finish(
|
||||||
ulang.get("npm.plugin_already", NAME=plugin_module_name, STATUS=ulang.get("npm.enable") if toggle else ulang.get("npm.disable")))
|
ulang.get("npm.plugin_already", NAME=plugin_module_name, STATUS=ulang.get("npm.enable") if toggle else ulang.get("npm.disable")))
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -210,7 +211,7 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
plugin_db.upsert(plugin)
|
plugin_db.upsert(plugin)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print(e)
|
print(e)
|
||||||
await toggle_plugin_global.finish(
|
await npm.finish(
|
||||||
ulang.get(
|
ulang.get(
|
||||||
"npm.toggle_failed",
|
"npm.toggle_failed",
|
||||||
NAME=plugin_module_name,
|
NAME=plugin_module_name,
|
||||||
@ -218,7 +219,7 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
ERROR=str(e))
|
ERROR=str(e))
|
||||||
)
|
)
|
||||||
|
|
||||||
await toggle_plugin_global.finish(
|
await npm.finish(
|
||||||
ulang.get(
|
ulang.get(
|
||||||
"npm.toggle_success",
|
"npm.toggle_success",
|
||||||
NAME=plugin_module_name,
|
NAME=plugin_module_name,
|
||||||
|
@ -1,18 +1,173 @@
|
|||||||
# 轻雪资源包管理器
|
# 轻雪资源包管理器
|
||||||
|
import os
|
||||||
|
|
||||||
|
import yaml
|
||||||
|
from nonebot import require
|
||||||
from nonebot.permission import SUPERUSER
|
from nonebot.permission import SUPERUSER
|
||||||
from nonebot_plugin_alconna import on_alconna, Alconna, Args, Subcommand, Arparma
|
|
||||||
|
|
||||||
from liteyuki.utils.ly_typing import T_Bot
|
from liteyuki.utils.language import get_user_lang
|
||||||
|
from liteyuki.utils.ly_typing import T_Bot, T_MessageEvent
|
||||||
|
from liteyuki.utils.message import Markdown as md
|
||||||
|
from liteyuki.utils.resource import (ResourceMetadata, add_resource_pack, change_priority, check_exist, check_status, get_loaded_resource_packs, get_resource_metadata, load_resources, remove_resource_pack)
|
||||||
|
|
||||||
list_rp = on_alconna(
|
require("nonebot_plugin_alconna")
|
||||||
aliases={"列出资源包", "资源包列表"},
|
from nonebot_plugin_alconna import Alconna, Args, on_alconna, Arparma, Subcommand
|
||||||
|
|
||||||
|
rpm = on_alconna(
|
||||||
|
aliases={"资源包"},
|
||||||
command=Alconna(
|
command=Alconna(
|
||||||
|
"rpm",
|
||||||
|
Subcommand(
|
||||||
"list",
|
"list",
|
||||||
Args["page", int, 1]["num", int, 10],
|
Args["page", int, 1]["num", int, 10],
|
||||||
|
alias=["ls", "列表", "列出"],
|
||||||
|
),
|
||||||
|
Subcommand(
|
||||||
|
"load",
|
||||||
|
Args["name", str],
|
||||||
|
alias=["安装"],
|
||||||
|
),
|
||||||
|
Subcommand(
|
||||||
|
"unload",
|
||||||
|
Args["name", str],
|
||||||
|
alias=["卸载"],
|
||||||
|
),
|
||||||
|
Subcommand(
|
||||||
|
"up",
|
||||||
|
Args["name", str],
|
||||||
|
alias=["上移"],
|
||||||
|
),
|
||||||
|
Subcommand(
|
||||||
|
"down",
|
||||||
|
Args["name", str],
|
||||||
|
alias=["下移"],
|
||||||
|
),
|
||||||
|
Subcommand(
|
||||||
|
"top",
|
||||||
|
Args["name", str],
|
||||||
|
alias=["置顶"],
|
||||||
|
),
|
||||||
|
Subcommand(
|
||||||
|
"reload",
|
||||||
|
alias=["重载"],
|
||||||
|
),
|
||||||
),
|
),
|
||||||
permission=SUPERUSER
|
permission=SUPERUSER
|
||||||
)
|
)
|
||||||
|
|
||||||
@list_rp.handle()
|
|
||||||
async def _(bot: T_Bot):
|
@rpm.handle()
|
||||||
pass
|
async def _(bot: T_Bot, event: T_MessageEvent, result: Arparma):
|
||||||
|
ulang = get_user_lang(str(event.user_id))
|
||||||
|
reply = ""
|
||||||
|
if result.subcommands.get("list"):
|
||||||
|
loaded_rps = get_loaded_resource_packs()
|
||||||
|
reply += f"{ulang.get('liteyuki.loaded_resources', NUM=len(loaded_rps))}\n"
|
||||||
|
for rp in loaded_rps:
|
||||||
|
btn_unload = md.btn_cmd(
|
||||||
|
ulang.get("npm.uninstall"),
|
||||||
|
f"rpm unload {rp.folder}"
|
||||||
|
)
|
||||||
|
btn_move_up = md.btn_cmd(
|
||||||
|
ulang.get("rpm.move_up"),
|
||||||
|
f"rpm up {rp.folder}"
|
||||||
|
)
|
||||||
|
btn_move_down = md.btn_cmd(
|
||||||
|
ulang.get("rpm.move_down"),
|
||||||
|
f"rpm down {rp.folder}"
|
||||||
|
)
|
||||||
|
btn_move_top = md.btn_cmd(
|
||||||
|
ulang.get("rpm.move_top"),
|
||||||
|
f"rpm top {rp.folder}"
|
||||||
|
)
|
||||||
|
# 添加新行
|
||||||
|
reply += (f"\n**{md.escape(rp.name)}**({md.escape(rp.folder)})\n\n"
|
||||||
|
f"> {btn_move_up} {btn_move_down} {btn_move_top} {btn_unload}\n\n***")
|
||||||
|
reply += f"\n\n{ulang.get('liteyuki.unloaded_resources')}\n"
|
||||||
|
loaded_folders = [rp.folder for rp in get_loaded_resource_packs()]
|
||||||
|
for folder in os.listdir("resources"):
|
||||||
|
if folder not in loaded_folders and os.path.exists(os.path.join("resources", folder, "metadata.yml")):
|
||||||
|
metadata = ResourceMetadata(
|
||||||
|
**yaml.load(
|
||||||
|
open(
|
||||||
|
os.path.join("resources", folder, "metadata.yml"),
|
||||||
|
encoding="utf-8"
|
||||||
|
),
|
||||||
|
Loader=yaml.FullLoader
|
||||||
|
)
|
||||||
|
)
|
||||||
|
metadata.folder = folder
|
||||||
|
metadata.path = os.path.join("resources", folder)
|
||||||
|
btn_load = md.btn_cmd(
|
||||||
|
ulang.get("npm.install"),
|
||||||
|
f"rpm load {metadata.folder}"
|
||||||
|
)
|
||||||
|
# 添加新行
|
||||||
|
reply += (f"\n**{md.escape(metadata.name)}**({md.escape(metadata.folder)})\n\n"
|
||||||
|
f"> {btn_load}\n\n***")
|
||||||
|
elif result.subcommands.get("load") or result.subcommands.get("unload"):
|
||||||
|
load = result.subcommands.get("load") is not None
|
||||||
|
rp_name = result.args.get("name")
|
||||||
|
r = False # 操作结果
|
||||||
|
if check_exist(rp_name):
|
||||||
|
if load != check_status(rp_name):
|
||||||
|
# 状态不同
|
||||||
|
if load:
|
||||||
|
r = add_resource_pack(rp_name)
|
||||||
|
else:
|
||||||
|
r = remove_resource_pack(rp_name)
|
||||||
|
rp_meta = get_resource_metadata(rp_name)
|
||||||
|
reply += ulang.get(
|
||||||
|
f"liteyuki.{'load' if load else 'unload'}_resource_{'success' if r else 'failed'}",
|
||||||
|
NAME=rp_meta.name
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
# 重复操作
|
||||||
|
reply += ulang.get(f"liteyuki.resource_already_{'load' if load else 'unload'}ed", NAME=rp_name)
|
||||||
|
else:
|
||||||
|
reply += ulang.get("liteyuki.resource_not_found", NAME=rp_name)
|
||||||
|
if r:
|
||||||
|
btn_reload = md.btn_cmd(
|
||||||
|
ulang.get("liteyuki.reload_resources"),
|
||||||
|
f"rpm reload"
|
||||||
|
)
|
||||||
|
reply += "\n" + ulang.get("liteyuki.need_reload", BTN=btn_reload)
|
||||||
|
elif result.subcommands.get("up") or result.subcommands.get("down") or result.subcommands.get("top"):
|
||||||
|
rp_name = result.args.get("name")
|
||||||
|
if result.subcommands.get("up"):
|
||||||
|
delta = -1
|
||||||
|
elif result.subcommands.get("down"):
|
||||||
|
delta = 1
|
||||||
|
else:
|
||||||
|
delta = 0
|
||||||
|
if check_exist(rp_name):
|
||||||
|
if check_status(rp_name):
|
||||||
|
r = change_priority(rp_name, delta)
|
||||||
|
reply += ulang.get(f"liteyuki.change_priority_{'success' if r else 'failed'}", NAME=rp_name)
|
||||||
|
if r:
|
||||||
|
btn_reload = md.btn_cmd(
|
||||||
|
ulang.get("liteyuki.reload_resources"),
|
||||||
|
f"rpm reload"
|
||||||
|
)
|
||||||
|
reply += "\n" + ulang.get("liteyuki.need_reload", BTN=btn_reload)
|
||||||
|
else:
|
||||||
|
reply += ulang.get("liteyuki.resource_not_found", NAME=rp_name)
|
||||||
|
else:
|
||||||
|
reply += ulang.get("liteyuki.resource_not_found", NAME=rp_name)
|
||||||
|
elif result.subcommands.get("reload"):
|
||||||
|
load_resources()
|
||||||
|
reply = ulang.get(
|
||||||
|
"liteyuki.reload_resources_success",
|
||||||
|
NUM=len(get_loaded_resource_packs())
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
btn_reload = md.btn_cmd(
|
||||||
|
ulang.get("liteyuki.reload_resources"),
|
||||||
|
f"rpm reload"
|
||||||
|
)
|
||||||
|
btn_list = md.btn_cmd(
|
||||||
|
ulang.get("liteyuki.list_resources"),
|
||||||
|
f"rpm list"
|
||||||
|
)
|
||||||
|
reply += f"{btn_list} {btn_reload}"
|
||||||
|
await md.send_md(reply, bot, event=event)
|
||||||
|
@ -90,7 +90,7 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
continue
|
continue
|
||||||
val = profile.dict()[key]
|
val = profile.dict()[key]
|
||||||
key_text = ulang.get(f"user.profile.{key}")
|
key_text = ulang.get(f"user.profile.{key}")
|
||||||
btn_set = md.cmd(ulang.get("user.profile.edit"), f"profile set {key}",
|
btn_set = md.btn_cmd(ulang.get("user.profile.edit"), f"profile set {key}",
|
||||||
enter=True if key in enter_attr else False)
|
enter=True if key in enter_attr else False)
|
||||||
reply += (f"\n**{key_text}** **{val}**\n"
|
reply += (f"\n**{key_text}** **{val}**\n"
|
||||||
f"\n> {ulang.get(f'user.profile.{key}.desc')}"
|
f"\n> {ulang.get(f'user.profile.{key}.desc')}"
|
||||||
@ -117,11 +117,11 @@ def get_profile_menu(key: str, ulang: Language) -> Optional[str]:
|
|||||||
reply = f"**{setting_name} {ulang.get('user.profile.settings')}**\n***\n"
|
reply = f"**{setting_name} {ulang.get('user.profile.settings')}**\n***\n"
|
||||||
if key == "lang":
|
if key == "lang":
|
||||||
for lang_code, lang_name in get_all_lang().items():
|
for lang_code, lang_name in get_all_lang().items():
|
||||||
btn_set = md.cmd(ulang.get("user.profile.set"), f"profile set {key} {lang_code}")
|
btn_set = md.btn_cmd(ulang.get("user.profile.set"), f"profile set {key} {lang_code}")
|
||||||
reply += f"\n{btn_set} | **{lang_name}** - {lang_code}\n***\n"
|
reply += f"\n{btn_set} | **{lang_name}** - {lang_code}\n***\n"
|
||||||
elif key == "timezone":
|
elif key == "timezone":
|
||||||
for tz in representative_timezones_list:
|
for tz in representative_timezones_list:
|
||||||
btn_set_tz = md.cmd(tz, f"profile set {key} {tz}")
|
btn_set_tz = md.btn_cmd(tz, f"profile set {key} {tz}")
|
||||||
reply += f"{btn_set_tz}\n"
|
reply += f"{btn_set_tz}\n"
|
||||||
return reply
|
return reply
|
||||||
|
|
||||||
|
@ -100,4 +100,23 @@ liteyuki.image_mode_off=关闭Markdown图片模式
|
|||||||
npm.page=第{PAGE}/{TOTAL}页
|
npm.page=第{PAGE}/{TOTAL}页
|
||||||
main.monitor.free=空闲
|
main.monitor.free=空闲
|
||||||
|
|
||||||
liteyuki.reload_resources_success=资源重载成功,共计{NUM}个资源包
|
liteyuki.invalid_command=无效的命令或参数
|
||||||
|
liteyuki.reload_resources=重载资源
|
||||||
|
liteyuki.list_resources=资源包列表
|
||||||
|
liteyuki.reload_resources_success=资源重载成功,共计 {NUM} 个资源包
|
||||||
|
liteyuki.loaded_resources=已加载 {NUM} 个资源包,按照优先级排序
|
||||||
|
liteyuki.unloaded_resources=未加载资源包
|
||||||
|
liteyuki.load_resource_success=资源包 {NAME} 加载成功
|
||||||
|
liteyuki.unload_resource_success=资源包 {NAME} 卸载成功
|
||||||
|
liteyuki.load_resource_failed=资源包 {NAME} 加载失败
|
||||||
|
liteyuki.unload_resource_failed=资源包 {NAME} 卸载失败
|
||||||
|
liteyuki.resource_not_found=资源包 {NAME} 不存在或不可操作
|
||||||
|
liteyuki.resource_already_loaded=资源包 {NAME} 已加载,请勿重复操作
|
||||||
|
liteyuki.resource_already_unloaded=资源包 {NAME} 已卸载,请勿重复操作
|
||||||
|
liteyuki.need_reload=请{BTN}重载以应用这些更新
|
||||||
|
liteyuki.dont_repeat=请勿重复操作
|
||||||
|
liteyuki.change_priority_success=资源包 {NAME} 优先级修改成功
|
||||||
|
liteyuki.change_priority_failed=资源包 {NAME} 优先级修改失败
|
||||||
|
rpm.move_up=上移
|
||||||
|
rpm.move_down=下移
|
||||||
|
rpm.move_top=置顶
|
@ -94,7 +94,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
<script type="text/javascript" src="https://cdn.jsdelivr.net/npm/echarts@4.3.0/dist/echarts.min.js"></script>
|
<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/echarts/4.3.0/echarts.min.js"></script>
|
||||||
|
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
|
@ -6,7 +6,7 @@ import sys
|
|||||||
import nonebot
|
import nonebot
|
||||||
|
|
||||||
__NAME__ = "LiteyukiBot"
|
__NAME__ = "LiteyukiBot"
|
||||||
__VERSION__ = "6.2.6" # 60201
|
__VERSION__ = "6.2.7" # 60201
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
|
|
||||||
|
@ -10,27 +10,6 @@ require("nonebot_plugin_htmlrender")
|
|||||||
|
|
||||||
from nonebot_plugin_htmlrender import *
|
from nonebot_plugin_htmlrender import *
|
||||||
|
|
||||||
|
|
||||||
# async def html2image(
|
|
||||||
# html: str,
|
|
||||||
# wait: int = 0,
|
|
||||||
# template_path: str = None,
|
|
||||||
# scale_factor: float = 2,
|
|
||||||
# **kwargs
|
|
||||||
# ) -> bytes:
|
|
||||||
# """
|
|
||||||
# Args:
|
|
||||||
# html: str: HTML 正文
|
|
||||||
# wait: 等待时间
|
|
||||||
# template_path: 模板路径
|
|
||||||
# scale_factor: 缩放因子,越高越清晰
|
|
||||||
# **kwargs: page 参数
|
|
||||||
#
|
|
||||||
# Returns:
|
|
||||||
#
|
|
||||||
# """
|
|
||||||
# return await html_to_pic(html, wait=wait, template_path=template_path, scale_factor=scale_factor)
|
|
||||||
|
|
||||||
async def template2html(
|
async def template2html(
|
||||||
template: str,
|
template: str,
|
||||||
templates: dict,
|
templates: dict,
|
||||||
@ -77,6 +56,7 @@ async def template2image(
|
|||||||
}
|
}
|
||||||
template_path = os.path.dirname(template)
|
template_path = os.path.dirname(template)
|
||||||
template_name = os.path.basename(template)
|
template_name = os.path.basename(template)
|
||||||
|
print(template_path, template_name)
|
||||||
|
|
||||||
if debug:
|
if debug:
|
||||||
raw_html = await template_to_html(
|
raw_html = await template_to_html(
|
||||||
|
@ -12,7 +12,6 @@ import nonebot
|
|||||||
from .config import config
|
from .config import config
|
||||||
from .data_manager import User, user_db
|
from .data_manager import User, user_db
|
||||||
|
|
||||||
_default_lang_code = "en"
|
|
||||||
_language_data = {
|
_language_data = {
|
||||||
"en": {
|
"en": {
|
||||||
"name": "English",
|
"name": "English",
|
||||||
@ -102,7 +101,7 @@ def load_from_dict(data: dict, lang_code: str):
|
|||||||
|
|
||||||
|
|
||||||
class Language:
|
class Language:
|
||||||
def __init__(self, lang_code: str = None, fallback_lang_code: str = "en"):
|
def __init__(self, lang_code: str = None, fallback_lang_code: str = "zh-CN"):
|
||||||
if lang_code is None:
|
if lang_code is None:
|
||||||
lang_code = config.get("default_language", get_default_lang())
|
lang_code = config.get("default_language", get_default_lang())
|
||||||
self.lang_code = lang_code
|
self.lang_code = lang_code
|
||||||
|
@ -185,7 +185,7 @@ class Markdown:
|
|||||||
# 等林文轩修好Lagrange.OneBot再说
|
# 等林文轩修好Lagrange.OneBot再说
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def cmd(name: str, cmd: str, reply: bool = False, enter: bool = True) -> str:
|
def btn_cmd(name: str, cmd: str, reply: bool = False, enter: bool = True) -> str:
|
||||||
"""生成点击回调按钮
|
"""生成点击回调按钮
|
||||||
Args:
|
Args:
|
||||||
name: 按钮显示内容
|
name: 按钮显示内容
|
||||||
@ -202,7 +202,7 @@ class Markdown:
|
|||||||
return f"[{name}](mqqapi://aio/inlinecmd?command={quote(cmd)}&reply={str(reply).lower()}&enter={str(enter).lower()})"
|
return f"[{name}](mqqapi://aio/inlinecmd?command={quote(cmd)}&reply={str(reply).lower()}&enter={str(enter).lower()})"
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def link(name: str, url: str) -> str:
|
def btn_link(name: str, url: str) -> str:
|
||||||
"""生成点击链接按钮
|
"""生成点击链接按钮
|
||||||
Args:
|
Args:
|
||||||
name: 链接显示内容
|
name: 链接显示内容
|
||||||
|
@ -1,21 +1,25 @@
|
|||||||
|
import json
|
||||||
import os
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
import nonebot
|
import nonebot
|
||||||
import yaml
|
import yaml
|
||||||
from typing import Any
|
|
||||||
|
|
||||||
from liteyuki.utils.data import LiteModel
|
from .data import LiteModel
|
||||||
|
from .language import get_default_lang
|
||||||
|
|
||||||
_loaded_resource_packs: list["ResourceMetadata"] = [] # 按照加载顺序排序
|
_loaded_resource_packs: list["ResourceMetadata"] = [] # 按照加载顺序排序
|
||||||
temp_resource_root = "data/liteyuki/resources"
|
temp_resource_root = "data/liteyuki/resources"
|
||||||
|
lang = get_default_lang()
|
||||||
|
|
||||||
|
|
||||||
class ResourceMetadata(LiteModel):
|
class ResourceMetadata(LiteModel):
|
||||||
name: str = "Unknown"
|
name: str = "Unknown"
|
||||||
version: str = "0.0.1"
|
version: str = "0.0.1"
|
||||||
description: str = "Unknown"
|
description: str = "Unknown"
|
||||||
path: str
|
path: str = ""
|
||||||
|
folder: str = ""
|
||||||
|
|
||||||
|
|
||||||
def load_resource_from_dir(path: str):
|
def load_resource_from_dir(path: str):
|
||||||
@ -36,10 +40,11 @@ def load_resource_from_dir(path: str):
|
|||||||
relative_path = os.path.relpath(os.path.join(root, file), path)
|
relative_path = os.path.relpath(os.path.join(root, file), path)
|
||||||
copy_file(os.path.join(root, file), os.path.join(temp_resource_root, relative_path))
|
copy_file(os.path.join(root, file), os.path.join(temp_resource_root, relative_path))
|
||||||
metadata["path"] = path
|
metadata["path"] = path
|
||||||
|
metadata["folder"] = os.path.basename(path)
|
||||||
if os.path.exists(os.path.join(path, "lang")):
|
if os.path.exists(os.path.join(path, "lang")):
|
||||||
from liteyuki.utils.language import load_from_dir
|
from liteyuki.utils.language import load_from_dir
|
||||||
load_from_dir(os.path.join(path, "lang"))
|
load_from_dir(os.path.join(path, "lang"))
|
||||||
_loaded_resource_packs.append(ResourceMetadata(**metadata))
|
_loaded_resource_packs.insert(0, ResourceMetadata(**metadata))
|
||||||
|
|
||||||
|
|
||||||
def get_path(path: str, abs_path: bool = False, default: Any = None) -> str | Any:
|
def get_path(path: str, abs_path: bool = False, default: Any = None) -> str | Any:
|
||||||
@ -76,7 +81,7 @@ def get_files(path: str, abs_path: bool = False) -> list[str]:
|
|||||||
|
|
||||||
def get_loaded_resource_packs() -> list[ResourceMetadata]:
|
def get_loaded_resource_packs() -> list[ResourceMetadata]:
|
||||||
"""
|
"""
|
||||||
获取已加载的资源包
|
获取已加载的资源包,优先级从前到后
|
||||||
Returns: 资源包列表
|
Returns: 资源包列表
|
||||||
"""
|
"""
|
||||||
return _loaded_resource_packs
|
return _loaded_resource_packs
|
||||||
@ -106,6 +111,117 @@ def load_resources():
|
|||||||
standard_resource_path = "liteyuki/resources"
|
standard_resource_path = "liteyuki/resources"
|
||||||
load_resource_from_dir(standard_resource_path)
|
load_resource_from_dir(standard_resource_path)
|
||||||
# 加载其他资源包
|
# 加载其他资源包
|
||||||
if os.path.exists("resources"):
|
|
||||||
for resource in os.listdir("resources"):
|
if not os.path.exists("resources"):
|
||||||
|
os.makedirs("resources", exist_ok=True)
|
||||||
|
|
||||||
|
if not os.path.exists("resources/index.json"):
|
||||||
|
json.dump([], open("resources/index.json", "w", encoding="utf-8"))
|
||||||
|
|
||||||
|
resource_index: list[str] = json.load(open("resources/index.json", "r", encoding="utf-8"))
|
||||||
|
resource_index.reverse() # 优先级高的后加载,但是排在前面
|
||||||
|
for resource in resource_index:
|
||||||
load_resource_from_dir(os.path.join("resources", resource))
|
load_resource_from_dir(os.path.join("resources", resource))
|
||||||
|
|
||||||
|
|
||||||
|
def check_status(name: str) -> bool:
|
||||||
|
"""
|
||||||
|
检查资源包是否已加载
|
||||||
|
Args:
|
||||||
|
name: 资源包名称,文件夹名
|
||||||
|
Returns: 是否已加载
|
||||||
|
"""
|
||||||
|
return name in [rp.folder for rp in get_loaded_resource_packs()]
|
||||||
|
|
||||||
|
|
||||||
|
def check_exist(name: str) -> bool:
|
||||||
|
"""
|
||||||
|
检查资源包文件夹是否存在于resources文件夹
|
||||||
|
Args:
|
||||||
|
name: 资源包名称,文件夹名
|
||||||
|
Returns: 是否存在
|
||||||
|
"""
|
||||||
|
return os.path.exists(os.path.join("resources", name, "metadata.yml"))
|
||||||
|
|
||||||
|
|
||||||
|
def add_resource_pack(name: str) -> bool:
|
||||||
|
"""
|
||||||
|
添加资源包,该操作仅修改index.json文件,不会加载资源包,要生效请重载资源
|
||||||
|
Args:
|
||||||
|
name: 资源包名称,文件夹名
|
||||||
|
Returns:
|
||||||
|
"""
|
||||||
|
if check_exist(name):
|
||||||
|
old_index: list[str] = json.load(open("resources/index.json", "r", encoding="utf-8"))
|
||||||
|
if name not in old_index:
|
||||||
|
old_index.append(name)
|
||||||
|
json.dump(old_index, open("resources/index.json", "w", encoding="utf-8"))
|
||||||
|
load_resource_from_dir(os.path.join("resources", name))
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
nonebot.logger.warning(lang.get("liteyuki.resource_loaded", name=name))
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
nonebot.logger.warning(lang.get("liteyuki.resource_not_exist", name=name))
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def remove_resource_pack(name: str) -> bool:
|
||||||
|
"""
|
||||||
|
移除资源包,该操作仅修改加载索引,要生效请重载资源
|
||||||
|
Args:
|
||||||
|
name: 资源包名称,文件夹名
|
||||||
|
Returns:
|
||||||
|
"""
|
||||||
|
if check_exist(name):
|
||||||
|
old_index: list[str] = json.load(open("resources/index.json", "r", encoding="utf-8"))
|
||||||
|
if name in old_index:
|
||||||
|
old_index.remove(name)
|
||||||
|
json.dump(old_index, open("resources/index.json", "w", encoding="utf-8"))
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
nonebot.logger.warning(lang.get("liteyuki.resource_not_loaded", name=name))
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
nonebot.logger.warning(lang.get("liteyuki.resource_not_exist", name=name))
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def change_priority(name: str, delta: int) -> bool:
|
||||||
|
"""
|
||||||
|
修改资源包优先级
|
||||||
|
Args:
|
||||||
|
name: 资源包名称,文件夹名
|
||||||
|
delta: 优先级变化,正数表示后移,负数表示前移,0表示移到最前
|
||||||
|
Returns:
|
||||||
|
"""
|
||||||
|
# 正数表示前移,负数表示后移
|
||||||
|
old_resource_list: list[str] = json.load(open("resources/index.json", "r", encoding="utf-8"))
|
||||||
|
new_resource_list = old_resource_list.copy()
|
||||||
|
if name in old_resource_list:
|
||||||
|
index = old_resource_list.index(name)
|
||||||
|
if 0 <= index + delta < len(old_resource_list):
|
||||||
|
new_index = index + delta
|
||||||
|
new_resource_list.remove(name)
|
||||||
|
new_resource_list.insert(new_index, name)
|
||||||
|
json.dump(new_resource_list, open("resources/index.json", "w", encoding="utf-8"))
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
nonebot.logger.warning("Priority change failed, out of range")
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
nonebot.logger.debug("Priority change failed, resource not loaded")
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def get_resource_metadata(name: str) -> ResourceMetadata:
|
||||||
|
"""
|
||||||
|
获取资源包元数据
|
||||||
|
Args:
|
||||||
|
name: 资源包名称,文件夹名
|
||||||
|
Returns:
|
||||||
|
"""
|
||||||
|
for rp in get_loaded_resource_packs():
|
||||||
|
if rp.folder == name:
|
||||||
|
return rp
|
||||||
|
return ResourceMetadata()
|
||||||
|
Loading…
Reference in New Issue
Block a user