app/liteyuki/liteyuki_main/core.py

399 lines
14 KiB
Python
Raw Normal View History

2024-04-01 12:29:04 +08:00
import base64
2024-04-10 23:47:10 +08:00
import time
2024-06-02 01:32:52 +08:00
from typing import Any, AnyStr
2024-03-29 14:58:24 +08:00
import nonebot
2024-03-31 06:26:01 +08:00
import pip
2024-04-10 23:47:10 +08:00
from nonebot import Bot, get_driver, require
from nonebot.adapters import satori
2024-04-24 15:07:57 +08:00
from nonebot.adapters.onebot.v11 import Message, escape, unescape
2024-04-01 12:29:04 +08:00
from nonebot.exception import MockApiException
from nonebot.internal.matcher import Matcher
2024-03-30 06:04:17 +08:00
from nonebot.permission import SUPERUSER
2024-05-12 02:47:14 +08:00
from liteyuki.utils.base.config import get_config, load_from_yaml
from liteyuki.utils.base.data_manager import StoredConfig, TempConfig, common_db
from liteyuki.utils.base.language import get_user_lang
from liteyuki.utils.base.ly_typing import T_Bot, T_MessageEvent
from liteyuki.utils.message.message import MarkdownMessage as md, broadcast_to_superusers
from liteyuki.utils.base.reloader import Reloader
from liteyuki.utils import event as event_utils, satori_utils
2024-04-12 01:07:53 +08:00
from .api import update_liteyuki
from ..utils.base.ly_function import get_function
2024-04-23 23:10:58 +08:00
require("nonebot_plugin_alconna")
require("nonebot_plugin_apscheduler")
2024-04-24 15:07:57 +08:00
from nonebot_plugin_alconna import UniMessage, on_alconna, Alconna, Args, Subcommand, Arparma, MultiVar
2024-04-12 01:07:53 +08:00
from nonebot_plugin_apscheduler import scheduler
2024-03-24 22:04:51 +08:00
2024-03-30 06:04:17 +08:00
driver = get_driver()
markdown_image = common_db.where_one(StoredConfig(), default=StoredConfig()).config.get("markdown_image", False)
2024-04-01 12:29:04 +08:00
@on_alconna(
command=Alconna(
"liteecho",
2024-05-09 16:37:50 +08:00
Args["text", str, ""],
),
permission=SUPERUSER
).handle()
2024-05-16 20:09:20 +08:00
# Satori OK
2024-04-24 15:12:26 +08:00
async def _(bot: T_Bot, matcher: Matcher, result: Arparma):
if result.main_args.get("text"):
await matcher.finish(Message(unescape(result.main_args.get("text"))))
else:
await matcher.finish(f"Hello, Liteyuki!\nBot {bot.self_id}")
@on_alconna(
aliases={"更新轻雪"},
command=Alconna(
"update-liteyuki"
2024-03-27 00:05:51 +08:00
),
permission=SUPERUSER
).handle()
2024-05-16 20:09:20 +08:00
# Satori OK
2024-03-27 00:05:51 +08:00
async def _(bot: T_Bot, event: T_MessageEvent):
# 使用git pull更新
2024-05-16 20:09:20 +08:00
ulang = get_user_lang(str(event.user.id if isinstance(event, satori.event.Event) else event.user_id))
2024-04-12 01:07:53 +08:00
success, logs = update_liteyuki()
2024-03-27 00:05:51 +08:00
reply = "Liteyuki updated!\n"
2024-03-27 07:57:04 +08:00
reply += f"```\n{logs}\n```\n"
btn_restart = md.btn_cmd(ulang.get("liteyuki.restart_now"), "reload-liteyuki")
2024-03-31 06:26:01 +08:00
pip.main(["install", "-r", "requirements.txt"])
2024-03-27 07:57:04 +08:00
reply += f"{ulang.get('liteyuki.update_restart', RESTART=btn_restart)}"
2024-03-31 06:22:53 +08:00
await md.send_md(reply, bot, event=event, at_sender=False)
2024-03-27 00:05:51 +08:00
@on_alconna(
aliases={"重启轻雪"},
command=Alconna(
"reload-liteyuki"
),
permission=SUPERUSER
).handle()
2024-05-16 20:09:20 +08:00
# Satori OK
2024-04-10 23:47:10 +08:00
async def _(matcher: Matcher, bot: T_Bot, event: T_MessageEvent):
await matcher.send("Liteyuki reloading")
temp_data = common_db.where_one(TempConfig(), default=TempConfig())
temp_data.data.update(
{
"reload" : True,
"reload_time" : time.time(),
"reload_bot_id" : bot.self_id,
"reload_session_type": event_utils.get_message_type(event),
"reload_session_id" : (event.group_id if event.message_type == "group" else event.user_id) if not isinstance(event,
satori.event.Event) else event.channel.id,
"delta_time" : 0
}
)
2024-04-27 02:20:44 +08:00
common_db.save(temp_data)
2024-04-10 23:47:10 +08:00
Reloader.reload(0)
2024-03-29 14:58:24 +08:00
@on_alconna(
aliases={"配置"},
command=Alconna(
"config",
Subcommand(
"set",
Args["key", str]["value", Any],
alias=["设置"],
),
Subcommand(
"get",
Args["key", str, None],
alias=["查询", "获取"]
),
Subcommand(
"remove",
Args["key", str],
alias=["删除"]
)
),
permission=SUPERUSER
).handle()
2024-05-16 20:09:20 +08:00
# Satori OK
async def _(result: Arparma, event: T_MessageEvent, bot: T_Bot, matcher: Matcher):
ulang = get_user_lang(str(event_utils.get_user_id(event)))
stored_config: StoredConfig = common_db.where_one(StoredConfig(), default=StoredConfig())
2024-03-29 14:58:24 +08:00
if result.subcommands.get("set"):
key, value = result.subcommands.get("set").args.get("key"), result.subcommands.get("set").args.get("value")
try:
value = eval(value)
except:
pass
stored_config.config[key] = value
2024-04-27 02:20:44 +08:00
common_db.save(stored_config)
await matcher.finish(f"{ulang.get('liteyuki.config_set_success', KEY=key, VAL=value)}")
2024-03-29 14:58:24 +08:00
elif result.subcommands.get("get"):
key = result.subcommands.get("get").args.get("key")
file_config = load_from_yaml("config.yml")
reply = f"{ulang.get('liteyuki.current_config')}"
if key:
reply += f"```dotenv\n{key}={file_config.get(key, stored_config.config.get(key))}\n```"
else:
reply = f"{ulang.get('liteyuki.current_config')}"
reply += f"\n{ulang.get('liteyuki.static_config')}\n```dotenv"
for k, v in file_config.items():
reply += f"\n{k}={v}"
reply += "\n```"
if len(stored_config.config) > 0:
reply += f"\n{ulang.get('liteyuki.stored_config')}\n```dotenv"
for k, v in stored_config.config.items():
reply += f"\n{k}={v} {type(v)}"
2024-03-29 14:58:24 +08:00
reply += "\n```"
2024-03-31 06:22:53 +08:00
await md.send_md(reply, bot, event=event)
elif result.subcommands.get("remove"):
key = result.subcommands.get("remove").args.get("key")
if key in stored_config.config:
stored_config.config.pop(key)
2024-04-27 02:20:44 +08:00
common_db.save(stored_config)
await matcher.finish(f"{ulang.get('liteyuki.config_remove_success', KEY=key)}")
else:
await matcher.finish(f"{ulang.get('liteyuki.invalid_command', TEXT=key)}")
2024-03-30 06:04:17 +08:00
@on_alconna(
aliases={"切换图片模式"},
command=Alconna(
"switch-image-mode"
),
permission=SUPERUSER
).handle()
2024-05-16 20:09:20 +08:00
# Satori OK
async def _(event: T_MessageEvent, matcher: Matcher):
2024-04-01 12:29:04 +08:00
global markdown_image
2024-04-01 11:46:47 +08:00
# 切换图片模式False以图片形式发送True以markdown形式发送
ulang = get_user_lang(str(event_utils.get_user_id(event)))
stored_config: StoredConfig = common_db.where_one(StoredConfig(), default=StoredConfig())
2024-04-01 11:46:47 +08:00
stored_config.config["markdown_image"] = not stored_config.config.get("markdown_image", False)
2024-04-01 12:29:04 +08:00
markdown_image = stored_config.config["markdown_image"]
2024-04-27 02:20:44 +08:00
common_db.save(stored_config)
2024-05-16 20:09:20 +08:00
await matcher.finish(
ulang.get("liteyuki.image_mode_on" if stored_config.config["markdown_image"] else "liteyuki.image_mode_off"))
2024-04-01 12:29:04 +08:00
@on_alconna(
command=Alconna(
"liteyuki-docs",
),
aliases={"轻雪文档"},
).handle()
2024-05-16 20:09:20 +08:00
# Satori OK
async def _(matcher: Matcher):
2024-04-12 01:07:53 +08:00
await matcher.finish("https://bot.liteyuki.icu/usage")
2024-04-01 12:29:04 +08:00
@on_alconna(
command=Alconna(
"/function",
Args["function", str]["args", MultiVar(str), ()],
),
permission=SUPERUSER
).handle()
async def _(result: Arparma, bot: T_Bot, event: T_MessageEvent, matcher: Matcher):
"""
调用轻雪函数
Args:
result:
bot:
event:
Returns:
"""
function_name = result.main_args.get("function")
args: tuple[str] = result.main_args.get("args", ())
_args = []
2024-06-02 01:32:52 +08:00
_kwargs = {
"USER_ID" : str(event.user_id),
"GROUP_ID": str(event.group_id) if event.message_type == "group" else "0",
"BOT_ID" : str(bot.self_id)
}
for arg in args:
arg = arg.replace("\\=", "EQUAL_SIGN")
if "=" in arg:
key, value = arg.split("=", 1)
value = unescape(value.replace("EQUAL_SIGN", "="))
try:
value = eval(value)
except:
value = value
_kwargs[key] = value
else:
_args.append(arg.replace("EQUAL_SIGN", "="))
ly_func = get_function(function_name)
2024-06-02 01:32:52 +08:00
ly_func.bot = bot if "BOT_ID" not in _kwargs else nonebot.get_bot(_kwargs["BOT_ID"])
ly_func.matcher = matcher
await ly_func(*tuple(_args), **_kwargs)
2024-04-21 14:15:12 +08:00
@on_alconna(
command=Alconna(
"/api",
2024-06-02 01:32:52 +08:00
Args["api", str]["args", MultiVar(AnyStr), ()],
2024-04-21 14:15:12 +08:00
),
permission=SUPERUSER
).handle()
async def _(result: Arparma, bot: T_Bot, event: T_MessageEvent, matcher: Matcher):
"""
调用API
Args:
result:
bot:
event:
Returns:
"""
api_name = result.main_args.get("api")
args: tuple[str] = result.main_args.get("args", ()) # 类似于url参数但每个参数间用空格分隔空格是%20
args_dict = {}
2024-04-24 15:07:57 +08:00
2024-04-21 14:15:12 +08:00
for arg in args:
key, value = arg.split("=", 1)
2024-06-02 01:32:52 +08:00
2024-04-24 15:07:57 +08:00
args_dict[key] = unescape(value.replace("%20", " "))
2024-04-21 14:15:12 +08:00
if api_name in need_user_id and "user_id" not in args_dict:
args_dict["user_id"] = str(event.user_id)
if api_name in need_group_id and "group_id" not in args_dict and event.message_type == "group":
args_dict["group_id"] = str(event.group_id)
2024-04-24 15:07:57 +08:00
if "message" in args_dict:
2024-06-02 01:32:52 +08:00
args_dict["message"] = Message(eval(args_dict["message"]))
if "messages" in args_dict:
args_dict["messages"] = Message(eval(args_dict["messages"]))
2024-04-24 15:07:57 +08:00
2024-04-21 14:15:12 +08:00
try:
result = await bot.call_api(api_name, **args_dict)
except Exception as e:
result = str(e)
args_show = "\n".join("- %s: %s" % (k, v) for k, v in args_dict.items())
2024-04-24 15:07:57 +08:00
print(f"API: {api_name}\n\nArgs: \n{args_show}\n\nResult: {result}")
2024-04-21 14:15:12 +08:00
await matcher.finish(f"API: {api_name}\n\nArgs: \n{args_show}\n\nResult: {result}")
# system hook
@Bot.on_calling_api # 图片模式检测
2024-04-01 12:29:04 +08:00
async def test_for_md_image(bot: T_Bot, api: str, data: dict):
# 截获大图发送转换为markdown发送
2024-05-16 20:09:20 +08:00
if api in ["send_msg", "send_private_msg", "send_group_msg"] and markdown_image and data.get(
"user_id") != bot.self_id:
2024-04-01 12:29:04 +08:00
if api == "send_msg" and data.get("message_type") == "private" or api == "send_private_msg":
session_type = "private"
session_id = data.get("user_id")
elif api == "send_msg" and data.get("message_type") == "group" or api == "send_group_msg":
session_type = "group"
session_id = data.get("group_id")
else:
return
if len(data.get("message", [])) == 1 and data["message"][0].get("type") == "image":
file: str = data["message"][0].data.get("file")
# file:// http:// base64://
if file.startswith("http"):
2024-05-16 20:09:20 +08:00
result = await md.send_md(await md.image_async(file), bot, message_type=session_type,
session_id=session_id)
2024-04-01 12:29:04 +08:00
elif file.startswith("file"):
file = file.replace("file://", "")
2024-05-16 20:09:20 +08:00
result = await md.send_image(open(file, "rb").read(), bot, message_type=session_type,
session_id=session_id)
2024-04-01 12:29:04 +08:00
elif file.startswith("base64"):
file_bytes = base64.b64decode(file.replace("base64://", ""))
result = await md.send_image(file_bytes, bot, message_type=session_type, session_id=session_id)
else:
return
raise MockApiException(result=result)
2024-04-01 11:46:47 +08:00
2024-03-30 06:04:17 +08:00
@driver.on_startup
async def on_startup():
temp_data = common_db.where_one(TempConfig(), default=TempConfig())
# 储存重启信息
2024-04-10 23:47:10 +08:00
if temp_data.data.get("reload", False):
delta_time = time.time() - temp_data.data.get("reload_time", 0)
temp_data.data["delta_time"] = delta_time
2024-04-27 02:20:44 +08:00
common_db.save(temp_data) # 更新数据
2024-03-30 06:04:17 +08:00
2024-03-30 06:04:17 +08:00
@driver.on_shutdown
async def on_shutdown():
pass
2024-04-10 23:47:10 +08:00
2024-04-10 23:47:10 +08:00
@driver.on_bot_connect
async def _(bot: T_Bot):
temp_data = common_db.where_one(TempConfig(), default=TempConfig())
2024-05-16 20:09:20 +08:00
if isinstance(bot, satori.Bot):
await satori_utils.user_infos.load_friends(bot)
# 用于重启计时
2024-04-10 23:47:10 +08:00
if temp_data.data.get("reload", False):
temp_data.data["reload"] = False
reload_bot_id = temp_data.data.get("reload_bot_id", 0)
if reload_bot_id != bot.self_id:
return
reload_session_type = temp_data.data.get("reload_session_type", "private")
reload_session_id = temp_data.data.get("reload_session_id", 0)
delta_time = temp_data.data.get("delta_time", 0)
2024-04-27 02:20:44 +08:00
common_db.save(temp_data) # 更新数据
if isinstance(bot, satori.Bot):
2024-05-16 20:09:20 +08:00
await bot.send_message(
channel_id=reload_session_id,
message="Liteyuki reloaded in %.2f s" % delta_time
)
else:
await bot.call_api(
"send_msg",
message_type=reload_session_type,
user_id=reload_session_id,
group_id=reload_session_id,
message="Liteyuki reloaded in %.2f s" % delta_time
)
2024-04-12 01:07:53 +08:00
2024-04-12 01:07:53 +08:00
# 每天4点更新
@scheduler.scheduled_job("cron", hour=4)
async def every_day_update():
2024-04-12 07:17:42 +08:00
if get_config("auto_update", default=True):
2024-04-12 01:07:53 +08:00
result, logs = update_liteyuki()
pip.main(["install", "-r", "requirements.txt"])
2024-04-12 01:07:53 +08:00
if result:
2024-04-12 01:15:05 +08:00
await broadcast_to_superusers(f"Liteyuki updated: ```\n{logs}\n```")
2024-04-12 01:07:53 +08:00
nonebot.logger.info(f"Liteyuki updated: {logs}")
Reloader.reload(5)
2024-04-12 01:07:53 +08:00
else:
2024-04-12 01:15:05 +08:00
nonebot.logger.info(logs)
2024-04-21 14:15:12 +08:00
# 需要用户id的api
2024-04-21 14:15:12 +08:00
need_user_id = (
"send_private_msg",
"send_msg",
"set_group_card",
"set_group_special_title",
"get_stranger_info",
"get_group_member_info"
2024-04-21 14:15:12 +08:00
)
need_group_id = (
"send_group_msg",
"send_msg",
"set_group_card",
"set_group_name",
"set_group_special_title",
"get_group_member_info",
"get_group_member_list",
"get_group_honor_info"
)