mirror of
https://github.com/LiteyukiStudio/LiteyukiBot.git
synced 2024-11-26 14:05:03 +08:00
feat: 添加插件启用和停用功能
This commit is contained in:
parent
2711d8844b
commit
73b593ff98
@ -14,10 +14,11 @@ __plugin_meta__ = PluginMetadata(
|
|||||||
homepage="https://github.com/snowykami/LiteyukiBot",
|
homepage="https://github.com/snowykami/LiteyukiBot",
|
||||||
extra={
|
extra={
|
||||||
"liteyuki_plugin": True,
|
"liteyuki_plugin": True,
|
||||||
|
"toggleable": False,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
auto_migrate()
|
auto_migrate() # 自动迁移数据库
|
||||||
|
|
||||||
sys_lang = get_system_lang()
|
sys_lang = get_system_lang()
|
||||||
nonebot.logger.info(sys_lang.get("main.current_language", LANG=sys_lang.get("language.name")))
|
nonebot.logger.info(sys_lang.get("main.current_language", LANG=sys_lang.get("language.name")))
|
||||||
|
@ -53,6 +53,6 @@ __plugin_meta__ = PluginMetadata(
|
|||||||
usage="",
|
usage="",
|
||||||
homepage="https://github.com/snowykami/LiteyukiBot",
|
homepage="https://github.com/snowykami/LiteyukiBot",
|
||||||
extra={
|
extra={
|
||||||
"liteyuki_plugin": True,
|
"liteyuki": True,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
@ -117,6 +117,6 @@ __plugin_meta__ = PluginMetadata(
|
|||||||
usage="",
|
usage="",
|
||||||
homepage="https://github.com/snowykami/LiteyukiBot",
|
homepage="https://github.com/snowykami/LiteyukiBot",
|
||||||
extra={
|
extra={
|
||||||
"liteyuki_plugin": True,
|
"liteyuki": True,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
@ -2,6 +2,7 @@ from nonebot.plugin import PluginMetadata
|
|||||||
from .manager import *
|
from .manager import *
|
||||||
from .installer import *
|
from .installer import *
|
||||||
from .helper import *
|
from .helper import *
|
||||||
|
from .permission import *
|
||||||
|
|
||||||
__author__ = "snowykami"
|
__author__ = "snowykami"
|
||||||
__plugin_meta__ = PluginMetadata(
|
__plugin_meta__ = PluginMetadata(
|
||||||
@ -16,6 +17,8 @@ __plugin_meta__ = PluginMetadata(
|
|||||||
type="application",
|
type="application",
|
||||||
homepage="https://github.com/snowykami/LiteyukiBot",
|
homepage="https://github.com/snowykami/LiteyukiBot",
|
||||||
extra={
|
extra={
|
||||||
"liteyuki_plugin": True,
|
"liteyuki": True,
|
||||||
|
"toggleable" : False,
|
||||||
|
"default_enable" : True,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
@ -1,8 +1,102 @@
|
|||||||
|
import json
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import aiofiles
|
||||||
|
import nonebot.plugin
|
||||||
|
|
||||||
from src.utils.data import Database, LiteModel
|
from src.utils.data import Database, LiteModel
|
||||||
from src.utils.data_manager import InstalledPlugin, plugin_db
|
from src.utils.data_manager import GroupChat, InstalledPlugin, User, group_db, plugin_db, user_db
|
||||||
|
from src.utils.typing import T_MessageEvent
|
||||||
|
|
||||||
LNPM_COMMAND_START = "lnpm"
|
LNPM_COMMAND_START = "lnpm"
|
||||||
|
|
||||||
|
|
||||||
|
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
|
||||||
|
|
||||||
|
|
||||||
|
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 get_plugin_default_enable(plugin_module_name: str) -> bool:
|
||||||
|
"""
|
||||||
|
获取插件默认启用状态,由插件定义,不存在则默认为启用
|
||||||
|
|
||||||
|
Args:
|
||||||
|
plugin_module_name (str): 插件模块名
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool: 插件默认状态
|
||||||
|
"""
|
||||||
|
return (nonebot.plugin.get_plugin(plugin_module_name).metadata.extra.get('default_enable', True)
|
||||||
|
if nonebot.plugin.get_plugin(plugin_module_name) and nonebot.plugin.get_plugin(plugin_module_name).metadata else True) \
|
||||||
|
if nonebot.plugin.get_plugin(plugin_module_name) else False
|
||||||
|
|
||||||
|
|
||||||
|
def get_plugin_current_enable(event: T_MessageEvent, plugin_module_name: str) -> bool:
|
||||||
|
"""
|
||||||
|
获取插件当前启用状态
|
||||||
|
|
||||||
|
Args:
|
||||||
|
plugin_module_name (str): 插件模块名
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool: 插件当前状态
|
||||||
|
"""
|
||||||
|
if event.message_type == "group":
|
||||||
|
session: GroupChat = group_db.first(GroupChat, 'group_id = ?', event.group_id, default=GroupChat(group_id=event.group_id))
|
||||||
|
else:
|
||||||
|
session: User = user_db.first(User, 'user_id = ?', event.user_id, default=User(user_id=event.user_id))
|
||||||
|
# 默认停用插件在启用列表内表示启用
|
||||||
|
# 默认停用插件不在启用列表内表示停用
|
||||||
|
# 默认启用插件在停用列表内表示停用
|
||||||
|
# 默认启用插件不在停用列表内表示启用
|
||||||
|
default_enable = get_plugin_default_enable(plugin_module_name)
|
||||||
|
if default_enable:
|
||||||
|
return plugin_module_name not in session.disabled_plugins
|
||||||
|
else:
|
||||||
|
return plugin_module_name in session.enabled_plugins
|
||||||
|
|
||||||
|
|
||||||
|
def get_plugin_can_be_toggle(plugin_module_name: str) -> bool:
|
||||||
|
"""
|
||||||
|
获取插件是否可以被启用/停用
|
||||||
|
|
||||||
|
Args:
|
||||||
|
plugin_module_name (str): 插件模块名
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool: 插件是否可以被启用/停用
|
||||||
|
"""
|
||||||
|
return nonebot.plugin.get_plugin(plugin_module_name).metadata.extra.get('toggleable', True) \
|
||||||
|
if nonebot.plugin.get_plugin(plugin_module_name) and nonebot.plugin.get_plugin(plugin_module_name).metadata else True
|
||||||
|
@ -22,7 +22,7 @@ from src.utils.data_manager import InstalledPlugin
|
|||||||
|
|
||||||
npm_alc = on_alconna(
|
npm_alc = on_alconna(
|
||||||
Alconna(
|
Alconna(
|
||||||
["lnpm", "插件管理"],
|
["npm", "插件"],
|
||||||
Subcommand(
|
Subcommand(
|
||||||
"update",
|
"update",
|
||||||
alias=["u"],
|
alias=["u"],
|
||||||
@ -42,29 +42,17 @@ npm_alc = on_alconna(
|
|||||||
Args["plugin_name", str],
|
Args["plugin_name", str],
|
||||||
alias=["rm", "移除", "卸载"],
|
alias=["rm", "移除", "卸载"],
|
||||||
),
|
),
|
||||||
|
Subcommand(
|
||||||
|
"list",
|
||||||
|
alias=["l", "ls", "列表"],
|
||||||
|
)
|
||||||
),
|
),
|
||||||
permission=SUPERUSER
|
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()
|
@npm_alc.handle()
|
||||||
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
||||||
@ -88,12 +76,15 @@ 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.button(ulang.get('npm.install'), 'lnpm install %s' % plugin.module_name)
|
btn_install = md.button(ulang.get('npm.install'), 'npm install %s' % plugin.module_name)
|
||||||
link_page = md.link(ulang.get('npm.homepage'), plugin.homepage)
|
link_page = md.link(ulang.get('npm.homepage'), plugin.homepage)
|
||||||
|
link_pypi = md.link(ulang.get('npm.pypi'), plugin.homepage)
|
||||||
|
|
||||||
reply += (f"\n{btn_install} **{plugin.name}**\n"
|
reply += (f"\n# **{plugin.name}**\n"
|
||||||
f"\n > **{plugin.desc}**\n"
|
f"\n> **{plugin.desc}**\n"
|
||||||
f"\n > {ulang.get('npm.author')}: {plugin.author} {link_page}\n\n***\n")
|
f"\n> {ulang.get('npm.author')}: {plugin.author}"
|
||||||
|
f"\n> *{md.escape(plugin.module_name)}*"
|
||||||
|
f"\n> {btn_install} {link_page} {link_pypi}\n\n***\n")
|
||||||
if len(rs) > max_show:
|
if len(rs) > max_show:
|
||||||
reply += f"\n{ulang.get('npm.too_many_results', HIDE_NUM=len(rs) - max_show)}"
|
reply += f"\n{ulang.get('npm.too_many_results', HIDE_NUM=len(rs) - max_show)}"
|
||||||
else:
|
else:
|
||||||
@ -102,14 +93,22 @@ 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)
|
||||||
await npm_alc.send(ulang.get("npm.installing", NAME=plugin_module_name))
|
await npm_alc.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:
|
||||||
|
await npm_alc.finish(ulang.get("npm.plugin_not_found", NAME=plugin_module_name))
|
||||||
|
|
||||||
|
homepage_btn = md.button(ulang.get('npm.homepage'), store_plugin.homepage)
|
||||||
if r:
|
if r:
|
||||||
nonebot.load_plugin(plugin_module_name) # 加载插件
|
|
||||||
|
r_load = nonebot.load_plugin(plugin_module_name) # 加载插件
|
||||||
installed_plugin = InstalledPlugin(module_name=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) # 查询数据库中是否已经安装
|
found_in_db_plugin = plugin_db.first(InstalledPlugin, "module_name = ?", plugin_module_name) # 查询数据库中是否已经安装
|
||||||
|
|
||||||
|
if r_load:
|
||||||
if found_in_db_plugin is None:
|
if found_in_db_plugin is None:
|
||||||
plugin_db.save(installed_plugin)
|
plugin_db.save(installed_plugin)
|
||||||
info = ulang.get('npm.install_success', NAME=store_plugin.name).replace('_', r'\\_') # markdown转义
|
info = ulang.get('npm.install_success', NAME=store_plugin.name).replace('_', r'\\_') # markdown转义
|
||||||
@ -122,7 +121,15 @@ async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
|||||||
else:
|
else:
|
||||||
await npm_alc.finish(ulang.get('npm.plugin_already_installed', NAME=store_plugin.name))
|
await npm_alc.finish(ulang.get('npm.plugin_already_installed', NAME=store_plugin.name))
|
||||||
else:
|
else:
|
||||||
info = ulang.get('npm.install_failed', NAME=plugin_module_name).replace('_', r'\\_')
|
info = ulang.get('npm.load_failed', NAME=plugin_module_name, HOMEPAGE=homepage_btn).replace('_', r'\\_')
|
||||||
|
await send_markdown(
|
||||||
|
f"{info}\n\n"
|
||||||
|
f"```\n{log}\n```\n",
|
||||||
|
bot,
|
||||||
|
event=event
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
info = ulang.get('npm.install_failed', NAME=plugin_module_name, HOMEPAGE=homepage_btn).replace('_', r'\\_')
|
||||||
await send_markdown(
|
await send_markdown(
|
||||||
f"{info}\n\n"
|
f"{info}\n\n"
|
||||||
f"```\n{log}\n```",
|
f"```\n{log}\n```",
|
||||||
@ -192,22 +199,7 @@ async def npm_search(keywords: list[str]) -> list[StorePlugin]:
|
|||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
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]:
|
def npm_install(plugin_module_name) -> tuple[bool, str]:
|
||||||
|
@ -1,41 +1,131 @@
|
|||||||
import nonebot.plugin
|
import nonebot.plugin
|
||||||
from nonebot import on_command
|
from nonebot import on_command
|
||||||
|
from nonebot.internal.matcher import Matcher
|
||||||
|
from nonebot.message import run_preprocessor
|
||||||
from nonebot.permission import SUPERUSER
|
from nonebot.permission import SUPERUSER
|
||||||
|
from nonebot_plugin_alconna import on_alconna, Alconna, Args, Arparma
|
||||||
|
|
||||||
from src.utils.data_manager import InstalledPlugin, plugin_db
|
from src.utils.data_manager import GroupChat, InstalledPlugin, User, group_db, plugin_db, user_db
|
||||||
from src.utils.message import Markdown as md, send_markdown
|
from src.utils.message import Markdown as md, send_markdown
|
||||||
from src.utils.permission import GROUP_ADMIN, GROUP_OWNER
|
from src.utils.permission import GROUP_ADMIN, GROUP_OWNER
|
||||||
from src.utils.typing import T_Bot, T_MessageEvent
|
from src.utils.typing import T_Bot, T_MessageEvent
|
||||||
from src.utils.language import get_user_lang
|
from src.utils.language import get_user_lang
|
||||||
|
from .common import get_plugin_can_be_toggle, get_plugin_current_enable, get_plugin_default_enable
|
||||||
|
from .installer import get_store_plugin
|
||||||
|
|
||||||
list_plugins = on_command("list-plugin", aliases={"列出插件", "插件列表"}, priority=0)
|
list_plugins = on_command("list-plugin", aliases={"列出插件", "插件列表"}, priority=0)
|
||||||
toggle_plugin = on_command("enable-plugin", aliases={"启用插件", "停用插件", "disable-plugin"}, priority=0, permission=SUPERUSER)
|
# toggle_plugin = on_command("enable-plugin", aliases={"启用插件", "停用插件", "disable-plugin"}, priority=0)
|
||||||
|
toggle_plugin = on_alconna(
|
||||||
|
Alconna(
|
||||||
|
['enable-plugin', 'disable-plugin'],
|
||||||
|
Args['plugin_name', str]['global', bool, False],
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@list_plugins.handle()
|
@list_plugins.handle()
|
||||||
async def _(event: T_MessageEvent, bot: T_Bot):
|
async def _(event: T_MessageEvent, bot: T_Bot):
|
||||||
lang = get_user_lang(str(event.user_id))
|
lang = get_user_lang(str(event.user_id))
|
||||||
reply = f"# {lang.get('npm.loaded_plugins')} | {lang.get('npm.total', TOTAL=len(nonebot.get_loaded_plugins()))} \n***"
|
reply = f"# {lang.get('npm.loaded_plugins')} | {lang.get('npm.total', TOTAL=len(nonebot.get_loaded_plugins()))} \n***\n"
|
||||||
for plugin in nonebot.get_loaded_plugins():
|
for plugin in nonebot.get_loaded_plugins():
|
||||||
# 检查是否有 metadata 属性
|
# 检查是否有 metadata 属性
|
||||||
btn_help = md.button(lang.get('npm.help'), f'help {plugin.name}', False)
|
# 添加帮助按钮
|
||||||
reply += f"\n{btn_help} "
|
btn_usage = md.button(lang.get('npm.usage'), f'help {plugin.name}', False)
|
||||||
|
store_plugin = await get_store_plugin(plugin.module_name)
|
||||||
|
|
||||||
|
if store_plugin:
|
||||||
|
btn_homepage = md.link(lang.get('npm.homepage'), store_plugin.homepage)
|
||||||
|
elif plugin.metadata and plugin.metadata.extra.get('liteyuki'):
|
||||||
|
btn_homepage = md.link(lang.get('npm.homepage'), "https://github.com/snowykami/LiteyukiBot")
|
||||||
|
else:
|
||||||
|
btn_homepage = lang.get('npm.homepage')
|
||||||
|
|
||||||
if plugin.metadata:
|
if plugin.metadata:
|
||||||
reply += (f"**{plugin.metadata.name}**\n"
|
reply += (f"\n**{md.escape(plugin.metadata.name)}**\n"
|
||||||
f"\n > {plugin.metadata.description}")
|
f"\n > {plugin.metadata.description}")
|
||||||
else:
|
else:
|
||||||
reply += (f"**{plugin.name}**\n"
|
reply += (f"**{md.escape(plugin.name)}**\n"
|
||||||
f"\n > {lang.get('npm.no_description')}")
|
f"\n > {lang.get('npm.no_description')}")
|
||||||
# if await GROUP_ADMIN(bot=bot, event=event) or await GROUP_OWNER(bot=bot, event=event) or await SUPERUSER(bot=bot, event=event):
|
|
||||||
|
reply += f"\n > {btn_usage} {btn_homepage}"
|
||||||
|
|
||||||
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):
|
||||||
btn_enable = md.button(lang.get('npm.enable'), f'enable-plugin {plugin.module_name}')
|
# 添加启用/停用插件按钮
|
||||||
btn_disable = md.button(lang.get('npm.disable'), f'disable-plugin {plugin.module_name}')
|
btn_toggle = lang.get('npm.disable') if plugin.metadata and not plugin.metadata.extra.get('toggleable') \
|
||||||
reply += f"\n > {btn_enable} {btn_disable}"
|
else md.button(lang.get('npm.disable'), f'enable-plugin {plugin.module_name}')
|
||||||
|
reply += f" {btn_toggle}"
|
||||||
|
|
||||||
if await SUPERUSER(bot, event):
|
if await SUPERUSER(bot, event):
|
||||||
plugin_in_database = plugin_db.first(InstalledPlugin, 'module_name = ?', plugin.module_name)
|
plugin_in_database = plugin_db.first(InstalledPlugin, 'module_name = ?', plugin.module_name)
|
||||||
|
# 添加移除插件
|
||||||
btn_remove = (
|
btn_remove = (
|
||||||
md.button(lang.get('npm.uninstall'), f'lnpm remove {plugin.module_name}')) if plugin_in_database else lang.get(
|
md.button(lang.get('npm.uninstall'), f'npm remove {plugin.module_name}')) if plugin_in_database else lang.get(
|
||||||
'npm.uninstall')
|
'npm.uninstall')
|
||||||
reply += f" {btn_remove}"
|
btn_toggle_global = lang.get('npm.disable') if plugin.metadata and not plugin.metadata.extra.get('toggleable') \
|
||||||
|
else md.button(lang.get('npm.disable_global'), f'disable-plugin {plugin.module_name} global')
|
||||||
|
reply += f" {btn_remove} {btn_toggle_global}"
|
||||||
|
|
||||||
reply += "\n\n***\n"
|
reply += "\n\n***\n"
|
||||||
await send_markdown(reply, bot, event=event)
|
await send_markdown(reply, bot, event=event)
|
||||||
|
|
||||||
|
|
||||||
|
@toggle_plugin.handle()
|
||||||
|
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot):
|
||||||
|
# 判断会话类型
|
||||||
|
ulang = get_user_lang(str(event.user_id))
|
||||||
|
plugin_module_name = result.args.get("plugin_name")
|
||||||
|
|
||||||
|
toggle = result.header_result == 'enable-plugin' # 判断是启用还是停用
|
||||||
|
current_enable = get_plugin_current_enable(event, plugin_module_name) # 获取插件当前状态
|
||||||
|
|
||||||
|
default_enable = get_plugin_default_enable(plugin_module_name) # 获取插件默认状态
|
||||||
|
can_be_toggled = get_plugin_can_be_toggle(plugin_module_name) # 获取插件是否可以被启用/停用
|
||||||
|
|
||||||
|
if not can_be_toggled:
|
||||||
|
await toggle_plugin.finish(ulang.get("npm.plugin_cannot_be_toggled", NAME=plugin_module_name))
|
||||||
|
|
||||||
|
if current_enable == toggle:
|
||||||
|
await toggle_plugin.finish(
|
||||||
|
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":
|
||||||
|
session = user_db.first(User, "user_id = ?", event.user_id, default=User(user_id=event.user_id))
|
||||||
|
else:
|
||||||
|
if await GROUP_ADMIN(bot, event) or await GROUP_OWNER(bot, event) or await SUPERUSER(bot, event):
|
||||||
|
session = group_db.first(GroupChat, "group_id = ?", event.group_id, default=GroupChat(group_id=event.group_id))
|
||||||
|
else:
|
||||||
|
return
|
||||||
|
# 启用 已停用的默认启用插件 将其从停用列表移除
|
||||||
|
# 启用 已停用的默认停用插件 将其放到启用列表
|
||||||
|
# 停用 已启用的默认启用插件 将其放到停用列表
|
||||||
|
# 停用 已启用的默认停用插件 将其从启用列表移除
|
||||||
|
try:
|
||||||
|
if toggle:
|
||||||
|
if default_enable:
|
||||||
|
session.disabled_plugins.remove(plugin_module_name)
|
||||||
|
else:
|
||||||
|
session.enabled_plugins.append(plugin_module_name)
|
||||||
|
else:
|
||||||
|
if default_enable:
|
||||||
|
session.disabled_plugins.append(plugin_module_name)
|
||||||
|
else:
|
||||||
|
session.enabled_plugins.remove(plugin_module_name)
|
||||||
|
except Exception as e:
|
||||||
|
await toggle_plugin.finish(
|
||||||
|
ulang.get(
|
||||||
|
"npm.toggle_failed",
|
||||||
|
NAME=plugin_module_name,
|
||||||
|
STATUS=ulang.get("npm.enable") if toggle else ulang.get("npm.disable"),
|
||||||
|
ERROR=str(e))
|
||||||
|
)
|
||||||
|
|
||||||
|
if event.message_type == "private":
|
||||||
|
user_db.save(session)
|
||||||
|
else:
|
||||||
|
group_db.save(session)
|
||||||
|
|
||||||
|
|
||||||
|
@run_preprocessor
|
||||||
|
async def _(event: T_MessageEvent, matcher: Matcher):
|
||||||
|
plugin = matcher.plugin
|
||||||
|
nonebot.logger.info(f"Plugin: {plugin.module_name}")
|
||||||
|
@ -9,6 +9,8 @@ __plugin_meta__ = PluginMetadata(
|
|||||||
usage="",
|
usage="",
|
||||||
homepage="https://github.com/snowykami/LiteyukiBot",
|
homepage="https://github.com/snowykami/LiteyukiBot",
|
||||||
extra={
|
extra={
|
||||||
"liteyuki_plugin": True,
|
"liteyuki": True,
|
||||||
|
"toggleable" : False,
|
||||||
|
"default_enable" : True,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
@ -15,8 +15,11 @@ data_manager.migrate_success=Model {NAME} migration successful
|
|||||||
npm.loaded_plugins=Loaded plugins
|
npm.loaded_plugins=Loaded plugins
|
||||||
npm.total=Total {TOTAL}
|
npm.total=Total {TOTAL}
|
||||||
npm.help=Help
|
npm.help=Help
|
||||||
|
npm.usage=Usage
|
||||||
npm.disable=Disable
|
npm.disable=Disable
|
||||||
|
npm.disable_global=DisableGlobal
|
||||||
npm.enable=Enable
|
npm.enable=Enable
|
||||||
|
npm.enable_global=EnableGlobal
|
||||||
npm.install=Install
|
npm.install=Install
|
||||||
npm.uninstall=Uninstall
|
npm.uninstall=Uninstall
|
||||||
npm.installing=Installing {NAME}...
|
npm.installing=Installing {NAME}...
|
||||||
@ -28,13 +31,20 @@ npm.search_result=Search results
|
|||||||
npm.search_no_result=No result found
|
npm.search_no_result=No result found
|
||||||
npm.too_many_results=Too many results found, {HIDE_NUM} hidden, please refine your search
|
npm.too_many_results=Too many results found, {HIDE_NUM} hidden, please refine your search
|
||||||
npm.install_success={NAME} installed successfully
|
npm.install_success={NAME} installed successfully
|
||||||
npm.install_failed={NAME} installation failed
|
npm.install_failed={NAME} installation failed, please check the console for more information, or visit the plugin's {HOMEPAGE}
|
||||||
|
npm.remove_success={NAME} uninstalled successfully
|
||||||
|
npm.remove_failed={NAME} uninstallation failed
|
||||||
|
npm.load_failed={NAME} loading failed, please check the console for more information, or visit the plugin's {HOMEPAGE}
|
||||||
|
npm.plugin_not_found={NAME} not found, please check the plugin's name
|
||||||
npm.plugin_not_installed={NAME} is not installed
|
npm.plugin_not_installed={NAME} is not installed
|
||||||
npm.plugin_already_installed={NAME} is already installed
|
npm.plugin_already_installed={NAME} is already installed
|
||||||
npm.author=Author
|
npm.author=Author
|
||||||
npm.homepage=Homepage
|
npm.homepage=Homepage
|
||||||
|
npm.pypi=PyPI
|
||||||
npm.next_page=Next
|
npm.next_page=Next
|
||||||
npm.prev_page=Prev
|
npm.prev_page=Prev
|
||||||
|
npm.plugin_cannot_be_toggled=This plugin {NAME} cannot be toggled
|
||||||
|
npm.toggle_failed=Failed to {STATUS} {NAME}: {ERROR}
|
||||||
|
|
||||||
user.profile_manager.query=Your {ATTR} is {VALUE}
|
user.profile_manager.query=Your {ATTR} is {VALUE}
|
||||||
user.profile_manager.set=Your {ATTR} has been set to {VALUE}
|
user.profile_manager.set=Your {ATTR} has been set to {VALUE}
|
@ -15,8 +15,11 @@ data_manager.migrate_success=データが正常に移行されました {NAME}
|
|||||||
npm.loaded_plugins=読み込まれたプラグイン
|
npm.loaded_plugins=読み込まれたプラグイン
|
||||||
npm.total=合計 {TOTAL}
|
npm.total=合計 {TOTAL}
|
||||||
npm.help=ヘルプ
|
npm.help=ヘルプ
|
||||||
|
npm.usage=使用法
|
||||||
npm.disable=無効
|
npm.disable=無効
|
||||||
|
npm.disable_global=グローバル無効
|
||||||
npm.enable=有効
|
npm.enable=有効
|
||||||
|
npm.enable_global=グローバル有効
|
||||||
npm.install=インストール
|
npm.install=インストール
|
||||||
npm.uninstall=アンインストール
|
npm.uninstall=アンインストール
|
||||||
npm.installing={NAME} インストール中
|
npm.installing={NAME} インストール中
|
||||||
@ -28,13 +31,20 @@ npm.search_result=検索結果
|
|||||||
npm.search_no_result=検索結果がありません
|
npm.search_no_result=検索結果がありません
|
||||||
npm.too_many_results=検索結果が多すぎます。{HIDE_NUM} 件の結果が非表示になりました
|
npm.too_many_results=検索結果が多すぎます。{HIDE_NUM} 件の結果が非表示になりました
|
||||||
npm.install_success={NAME} が正常にインストールされました
|
npm.install_success={NAME} が正常にインストールされました
|
||||||
npm.install_failed={NAME} のインストールに失敗しました
|
npm.install_failed={NAME} のインストールに失敗しました, 詳細はログを参照してください, またはプラグインの作者に連絡してください{HOMEPAGE}
|
||||||
|
npm.remove_success={NAME} が正常にアンインストールされました
|
||||||
|
npm.remove_failed={NAME} のアンインストールに失敗しました
|
||||||
|
npm.load_failed={NAME} の読み込みに失敗しました,詳細はログを参照してください, またはプラグインの作者に連絡してください{HOMEPAGE}
|
||||||
|
npm.plugin_not_found={NAME} は見つかりません,スペルをチェックしてください
|
||||||
npm.plugin_not_installed={NAME} はインストールされていません
|
npm.plugin_not_installed={NAME} はインストールされていません
|
||||||
npm.plugin_already_installed={NAME} は既にインストールされています
|
npm.plugin_already_installed={NAME} は既にインストールされています
|
||||||
npm.author=著者
|
npm.author=著者
|
||||||
npm.homepage=ホームページ
|
npm.homepage=ホームページ
|
||||||
|
npm.pypi=PyPI
|
||||||
npm.next_page=次のページ
|
npm.next_page=次のページ
|
||||||
npm.prev_page=前のページ
|
npm.prev_page=前のページ
|
||||||
|
npm.plugin_cannot_be_toggled=このプラグイン {NAME} は無効にできません
|
||||||
|
npm.toggle_failed=プラグイン {NAME} の{STATUS}切り替えに失敗しました:{ERROR}
|
||||||
|
|
||||||
user.profile_manager.query=あなたのプロファイル情報 {ATTR} は {VALUE} です
|
user.profile_manager.query=あなたのプロファイル情報 {ATTR} は {VALUE} です
|
||||||
user.profile_manager.set=あなたのプロファイル情報 {ATTR} が {VALUE} に設定されました
|
user.profile_manager.set=あなたのプロファイル情報 {ATTR} が {VALUE} に設定されました
|
@ -15,8 +15,11 @@ data_manager.migrate_success=数据模型{NAME}迁移成功
|
|||||||
npm.loaded_plugins=已加载插件
|
npm.loaded_plugins=已加载插件
|
||||||
npm.total=总计 {TOTAL}
|
npm.total=总计 {TOTAL}
|
||||||
npm.help=帮助
|
npm.help=帮助
|
||||||
|
npm.usage=用法
|
||||||
npm.disable=停用
|
npm.disable=停用
|
||||||
|
npm.disable_global=全局停用
|
||||||
npm.enable=启用
|
npm.enable=启用
|
||||||
|
npm.enable_global=全局启用
|
||||||
npm.install=安装
|
npm.install=安装
|
||||||
npm.uninstall=卸载
|
npm.uninstall=卸载
|
||||||
npm.installing=正在安装 {NAME}
|
npm.installing=正在安装 {NAME}
|
||||||
@ -26,17 +29,23 @@ npm.store_update_success=插件商店数据更新成功
|
|||||||
npm.store_update_failed=插件商店数据更新失败
|
npm.store_update_failed=插件商店数据更新失败
|
||||||
npm.search_result=搜索结果
|
npm.search_result=搜索结果
|
||||||
npm.search_no_result=无搜索结果
|
npm.search_no_result=无搜索结果
|
||||||
npm.too_many_results=搜索结果过多,{HIDE_NUM}已隐藏,请限制关键字
|
npm.too_many_results=内容过多,{HIDE_NUM}项已隐藏,请限制关键字搜索
|
||||||
npm.install_success={NAME} 安装成功
|
npm.install_success={NAME} 安装成功
|
||||||
npm.install_failed={NAME} 安装失败
|
npm.install_failed={NAME} 安装失败,请查看日志获取详细信息,如不能解决,请访问{HOMEPAGE}寻求帮助
|
||||||
npm.remove_success={NAME} 卸载成功
|
npm.remove_success={NAME} 卸载成功
|
||||||
npm.remove_failed={NAME} 卸载失败
|
npm.remove_failed={NAME} 卸载失败
|
||||||
|
npm.load_failed={NAME} 加载失败,请在控制台查看详细信息,检查依赖或配置是否正确,如不能解决,请访问{HOMEPAGE}寻求帮助
|
||||||
|
npm.plugin_not_found=未在商店中找到 {NAME},请尝试更新商店信息或检查拼写
|
||||||
npm.plugin_not_installed={NAME} 未安装
|
npm.plugin_not_installed={NAME} 未安装
|
||||||
npm.plugin_already_installed={NAME} 已安装,请勿重复安装
|
npm.plugin_already_installed={NAME} 已安装,请勿重复安装
|
||||||
npm.author=作者
|
npm.author=作者
|
||||||
npm.homepage=项目主页
|
npm.homepage=主页
|
||||||
|
npm.pypi=PyPI
|
||||||
npm.next_page=下一页
|
npm.next_page=下一页
|
||||||
npm.prev_page=上一页
|
npm.prev_page=上一页
|
||||||
|
npm.plugin_cannot_be_toggled=插件 {NAME} 无法被启用或停用
|
||||||
|
npm.plugin_already=插件 {NAME} 已经是 {STATUS} 状态,无需重复操作
|
||||||
|
npm.toggle_failed=插件 {NAME} {STATUS} 失败: {ERROR}
|
||||||
|
|
||||||
user.profile_manager.query=你的个人信息 {ATTR} 为 {VALUE}
|
user.profile_manager.query=你的个人信息 {ATTR} 为 {VALUE}
|
||||||
user.profile_manager.set=你的个人信息 {ATTR} 已设置为 {VALUE}
|
user.profile_manager.set=你的个人信息 {ATTR} 已设置为 {VALUE}
|
@ -87,7 +87,7 @@ class Database(BaseORMAdapter):
|
|||||||
list : 'TEXT'
|
list : 'TEXT'
|
||||||
}
|
}
|
||||||
|
|
||||||
DEFAULT_TYPE = {
|
DEFAULT_VALUE = {
|
||||||
'TEXT' : '',
|
'TEXT' : '',
|
||||||
'INTEGER': 0,
|
'INTEGER': 0,
|
||||||
'REAL' : 0.0
|
'REAL' : 0.0
|
||||||
@ -95,6 +95,8 @@ class Database(BaseORMAdapter):
|
|||||||
|
|
||||||
FOREIGNID = 'FOREIGNID'
|
FOREIGNID = 'FOREIGNID'
|
||||||
JSON = 'JSON'
|
JSON = 'JSON'
|
||||||
|
LIST = 'LIST'
|
||||||
|
DICT = 'DICT'
|
||||||
ID = '$ID'
|
ID = '$ID'
|
||||||
|
|
||||||
def __init__(self, db_name: str):
|
def __init__(self, db_name: str):
|
||||||
@ -131,28 +133,34 @@ class Database(BaseORMAdapter):
|
|||||||
table_fields = self.cursor.fetchall()
|
table_fields = self.cursor.fetchall()
|
||||||
table_fields = [field[1] for field in table_fields]
|
table_fields = [field[1] for field in table_fields]
|
||||||
|
|
||||||
raw_fields = model.__annotations__.keys()
|
raw_fields, raw_types = zip(*model.__annotations__.items())
|
||||||
# 获取模型字段,若有模型则添加FOREIGNID前缀,若为BaseIterable则添加JSON前缀,用多行if判断
|
# 获取模型字段,若有模型则添加FOREIGNID前缀,若为BaseIterable则添加JSON前缀,用多行if判断
|
||||||
model_fields = []
|
model_fields = []
|
||||||
model_types = []
|
model_types = []
|
||||||
for field in raw_fields:
|
for field, r_type in zip(raw_fields, raw_types):
|
||||||
if isinstance(model.__annotations__[field], type(LiteModel)):
|
if isinstance(r_type, type(LiteModel)):
|
||||||
model_fields.append(f'{self.FOREIGNID}{field}')
|
model_fields.append(f'{self.FOREIGNID}{field}')
|
||||||
model_types.append('TEXT')
|
model_types.append('TEXT')
|
||||||
elif isinstance(model.__annotations__[field], types.GenericAlias):
|
elif isinstance(r_type, list):
|
||||||
|
model_fields.append(f'{self.LIST}{field}')
|
||||||
|
model_types.append('TEXT')
|
||||||
|
elif isinstance(r_type, dict):
|
||||||
|
model_fields.append(f'{self.DICT}{field}')
|
||||||
|
model_types.append('TEXT')
|
||||||
|
elif isinstance(r_type, types.GenericAlias):
|
||||||
model_fields.append(f'{self.JSON}{field}')
|
model_fields.append(f'{self.JSON}{field}')
|
||||||
model_types.append('TEXT')
|
model_types.append('TEXT')
|
||||||
else:
|
else:
|
||||||
model_fields.append(field)
|
model_fields.append(field)
|
||||||
model_types.append(self.type_map.get(model.__annotations__[field], 'TEXT'))
|
model_types.append(self.type_map.get(r_type, 'TEXT'))
|
||||||
|
|
||||||
# 检测新字段
|
# 检测新字段或字段类型是否有变化,有则增删字段,已经加了前缀类型
|
||||||
for field, type_ in zip(model_fields, model_types):
|
for field, type_, r_type in zip(model_fields, model_types, raw_types):
|
||||||
if field not in table_fields:
|
if field not in table_fields:
|
||||||
nonebot.logger.debug(f'ALTER TABLE {table_name} ADD COLUMN {field} {type_}')
|
nonebot.logger.debug(f'ALTER TABLE {table_name} ADD COLUMN {field} {type_}')
|
||||||
self.cursor.execute(f'ALTER TABLE {table_name} ADD COLUMN {field} {type_}')
|
self.cursor.execute(f'ALTER TABLE {table_name} ADD COLUMN {field} {type_}')
|
||||||
# 在原有的行中添加新字段对应类型的默认值,从DEFAULT_TYPE中获取
|
# 在原有的行中添加新字段对应类型的默认值,从DEFAULT_TYPE中获取
|
||||||
self.cursor.execute(f'UPDATE {table_name} SET {field} = ? WHERE {field} IS NULL', (self.DEFAULT_TYPE.get(type_, ""),))
|
self.cursor.execute(f'UPDATE {table_name} SET {field} = ? WHERE {field} IS NULL', (self.DEFAULT_VALUE.get(type_, ""),))
|
||||||
|
|
||||||
# 检测多余字段,除了id字段
|
# 检测多余字段,除了id字段
|
||||||
for field in table_fields:
|
for field in table_fields:
|
||||||
@ -161,7 +169,7 @@ class Database(BaseORMAdapter):
|
|||||||
self.cursor.execute(f'ALTER TABLE {table_name} DROP COLUMN {field}')
|
self.cursor.execute(f'ALTER TABLE {table_name} DROP COLUMN {field}')
|
||||||
|
|
||||||
self.conn.commit()
|
self.conn.commit()
|
||||||
nonebot.logger.success(f'Table {table_name} migrated successfully')
|
nonebot.logger.debug(f'Table {table_name} migrated successfully')
|
||||||
|
|
||||||
def save(self, *models: LiteModel) -> int | tuple:
|
def save(self, *models: LiteModel) -> int | tuple:
|
||||||
"""存储数据,检查id字段,如果有id字段则更新,没有则插入
|
"""存储数据,检查id字段,如果有id字段则更新,没有则插入
|
||||||
@ -326,7 +334,14 @@ class Database(BaseORMAdapter):
|
|||||||
new_d[k.replace(self.FOREIGNID, '')] = load(
|
new_d[k.replace(self.FOREIGNID, '')] = load(
|
||||||
dict(self.cursor.execute(f'SELECT * FROM {v.split(":", 2)[1]} WHERE id = ?', (v.split(":", 2)[2],)).fetchone()))
|
dict(self.cursor.execute(f'SELECT * FROM {v.split(":", 2)[1]} WHERE id = ?', (v.split(":", 2)[2],)).fetchone()))
|
||||||
elif k.startswith(self.JSON):
|
elif k.startswith(self.JSON):
|
||||||
|
if v == '': v = '[]'
|
||||||
new_d[k.replace(self.JSON, '')] = load(json.loads(v))
|
new_d[k.replace(self.JSON, '')] = load(json.loads(v))
|
||||||
|
elif k.startswith(self.LIST):
|
||||||
|
if v == '': v = '[]'
|
||||||
|
new_d[k.replace(self.LIST, '')] = load(json.loads(v))
|
||||||
|
elif k.startswith(self.DICT):
|
||||||
|
if v == '': v = '{}'
|
||||||
|
new_d[k.replace(self.DICT, '')] = load(json.loads(v))
|
||||||
else:
|
else:
|
||||||
new_d[k] = v
|
new_d[k] = v
|
||||||
elif isinstance(d, list | tuple | set):
|
elif isinstance(d, list | tuple | set):
|
||||||
|
@ -5,6 +5,7 @@ from src.utils.data import LiteModel, Database as DB
|
|||||||
DATA_PATH = "data/liteyuki"
|
DATA_PATH = "data/liteyuki"
|
||||||
|
|
||||||
user_db = DB(os.path.join(DATA_PATH, 'users.ldb'))
|
user_db = DB(os.path.join(DATA_PATH, 'users.ldb'))
|
||||||
|
group_db = DB(os.path.join(DATA_PATH, 'groups.ldb'))
|
||||||
plugin_db = DB(os.path.join(DATA_PATH, 'plugins.ldb'))
|
plugin_db = DB(os.path.join(DATA_PATH, 'plugins.ldb'))
|
||||||
|
|
||||||
|
|
||||||
@ -12,6 +13,16 @@ class User(LiteModel):
|
|||||||
user_id: str
|
user_id: str
|
||||||
username: str = ""
|
username: str = ""
|
||||||
lang: str = "en"
|
lang: str = "en"
|
||||||
|
enabled_plugins: list[str] = []
|
||||||
|
disabled_plugins: list[str] = []
|
||||||
|
|
||||||
|
|
||||||
|
class GroupChat(LiteModel):
|
||||||
|
# Group是一个关键字,所以这里用GroupChat
|
||||||
|
group_id: str
|
||||||
|
group_name: str = ""
|
||||||
|
enabled_plugins: list[str] = []
|
||||||
|
disabled_plugins: list[str] = []
|
||||||
|
|
||||||
|
|
||||||
class InstalledPlugin(LiteModel):
|
class InstalledPlugin(LiteModel):
|
||||||
@ -20,4 +31,5 @@ class InstalledPlugin(LiteModel):
|
|||||||
|
|
||||||
def auto_migrate():
|
def auto_migrate():
|
||||||
user_db.auto_migrate(User)
|
user_db.auto_migrate(User)
|
||||||
|
group_db.auto_migrate(GroupChat)
|
||||||
plugin_db.auto_migrate(InstalledPlugin)
|
plugin_db.auto_migrate(InstalledPlugin)
|
||||||
|
@ -76,7 +76,7 @@ async def send_markdown(markdown: str, bot: T_Bot, *, message_type: str = None,
|
|||||||
class Markdown:
|
class Markdown:
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def button(name: str, cmd: str, reply: bool = False, enter: bool = True) -> str:
|
def button(name: str, cmd: str, reply: bool = False, enter: bool = True) -> str:
|
||||||
"""生成点击按钮
|
"""生成点击回调按钮
|
||||||
Args:
|
Args:
|
||||||
name: 按钮显示内容
|
name: 按钮显示内容
|
||||||
cmd: 发送的命令,已在函数内url编码,不需要再次编码
|
cmd: 发送的命令,已在函数内url编码,不需要再次编码
|
||||||
@ -91,7 +91,7 @@ class Markdown:
|
|||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def link(name: str, url: str) -> str:
|
def link(name: str, url: str) -> str:
|
||||||
"""生成链接
|
"""生成点击链接按钮
|
||||||
Args:
|
Args:
|
||||||
name: 链接显示内容
|
name: 链接显示内容
|
||||||
url: 链接地址
|
url: 链接地址
|
||||||
@ -101,3 +101,18 @@ class Markdown:
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
return f"[🔗{name}]({url})"
|
return f"[🔗{name}]({url})"
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def escape(text: str) -> str:
|
||||||
|
"""转义特殊字符
|
||||||
|
Args:
|
||||||
|
text: 需要转义的文本,请勿直接把整个markdown文本传入,否则会转义掉所有字符
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
转义后的文本
|
||||||
|
|
||||||
|
"""
|
||||||
|
chars = "*[]()~_-`>#+-=|{}.!"
|
||||||
|
for char in chars:
|
||||||
|
text = text.replace(char, f"\\\\{char}")
|
||||||
|
return text
|
||||||
|
Loading…
Reference in New Issue
Block a user