mirror of
https://github.com/zhenxun-org/zhenxun_bot.git
synced 2025-12-15 14:22:55 +08:00
modified: basic_plugins/admin_bot_manage/admin_config.py modified: basic_plugins/admin_bot_manage/custom_welcome_message.py modified: basic_plugins/admin_bot_manage/timing_task.py modified: basic_plugins/apscheduler/__init__.py modified: basic_plugins/ban/__init__.py modified: basic_plugins/ban/data_source.py modified: basic_plugins/chat_history/chat_message.py modified: basic_plugins/chat_history/chat_message_handle.py modified: basic_plugins/group_handle/__init__.py modified: basic_plugins/hooks/_utils.py modified: basic_plugins/hooks/ban_hook.py modified: basic_plugins/hooks/chkdsk_hook.py modified: basic_plugins/init_plugin_config/__init__.py deleted: basic_plugins/init_plugin_config/init_group_manager.py modified: basic_plugins/invite_manager/__init__.py new file: basic_plugins/invite_manager/utils.py modified: basic_plugins/nickname.py modified: basic_plugins/plugin_shop/__init__.py modified: basic_plugins/plugin_shop/data_source.py modified: basic_plugins/scripts.py modified: basic_plugins/shop/__init__.py modified: basic_plugins/shop/buy.py modified: basic_plugins/shop/gold.py modified: basic_plugins/shop/my_props/__init__.py deleted: basic_plugins/shop/reset_today_gold.py modified: basic_plugins/shop/shop_handle/__init__.py modified: basic_plugins/shop/shop_handle/data_source.py modified: basic_plugins/shop/use/__init__.py modified: basic_plugins/shop/use/data_source.py modified: basic_plugins/super_cmd/__init__.py modified: basic_plugins/super_cmd/bot_friend_group.py modified: basic_plugins/super_cmd/clear_data.py modified: basic_plugins/super_cmd/exec_sql.py modified: basic_plugins/super_cmd/manager_group.py modified: basic_plugins/super_cmd/reload_setting.py modified: basic_plugins/super_cmd/set_admin_permissions.py deleted: basic_plugins/super_cmd/super_task_switch.py modified: basic_plugins/super_cmd/update_friend_group_info.py modified: basic_plugins/super_help/__init__.py modified: basic_plugins/update_info.py modified: configs/config.py modified: configs/utils/__init__.py modified: models/bag_user.py modified: models/ban_user.py modified: models/chat_history.py modified: models/friend_user.py modified: models/goods_info.py modified: models/group_info.py modified: models/group_member_info.py modified: models/level_user.py modified: models/sign_group_user.py modified: models/user_shop_gold_log.py modified: plugins/aconfig/__init__.py modified: plugins/ai/__init__.py modified: plugins/ai/data_source.py modified: plugins/bilibili_sub/__init__.py modified: plugins/bilibili_sub/data_source.py modified: plugins/bilibili_sub/model.py modified: plugins/black_word/__init__.py modified: plugins/black_word/model.py modified: plugins/black_word/utils.py modified: plugins/bt/data_source.py modified: plugins/genshin/almanac/__init__.py modified: plugins/genshin/material_remind/__init__.py modified: plugins/genshin/query_user/_models/__init__.py modified: plugins/genshin/query_user/_utils/__init__.py modified: plugins/genshin/query_user/bind/__init__.py modified: plugins/genshin/query_user/genshin_sign/__init__.py modified: plugins/genshin/query_user/genshin_sign/data_source.py modified: plugins/genshin/query_user/genshin_sign/init_task.py modified: plugins/genshin/query_user/mihoyobbs_sign/__init__.py modified: plugins/genshin/query_user/query_memo/__init__.py modified: plugins/genshin/query_user/query_memo/data_source.py modified: plugins/genshin/query_user/query_role/__init__.py modified: plugins/genshin/query_user/query_role/data_source.py modified: plugins/genshin/query_user/reset_today_query_user_data/__init__.py modified: plugins/genshin/query_user/resin_remind/__init__.py modified: plugins/genshin/query_user/resin_remind/init_task.py modified: plugins/gold_redbag/model.py modified: plugins/image_management/send_image/__init__.py modified: plugins/my_info/__init__.py modified: plugins/open_cases/models/buff_prices.py modified: plugins/open_cases/models/open_cases_user.py modified: plugins/open_cases/open_cases_c.py modified: plugins/open_cases/utils.py modified: plugins/parse_bilibili_json.py modified: plugins/pid_search.py modified: plugins/pix_gallery/__init__.py modified: plugins/pix_gallery/_data_source.py modified: plugins/pix_gallery/_model/omega_pixiv_illusts.py modified: plugins/pix_gallery/_model/pixiv.py modified: plugins/pix_gallery/_model/pixiv_keyword_user.py modified: plugins/pix_gallery/pix_add_keyword.py modified: plugins/pix_gallery/pix_pass_del_keyword.py modified: plugins/pix_gallery/pix_show_info.py modified: plugins/pix_gallery/pix_update.py modified: plugins/pixiv_rank_search/data_source.py modified: plugins/poke/__init__.py modified: plugins/russian/__init__.py modified: plugins/russian/data_source.py modified: plugins/russian/model.py modified: plugins/send_dinggong_voice/__init__.py modified: plugins/send_setu_/_model.py modified: plugins/send_setu_/send_setu/__init__.py modified: plugins/send_setu_/send_setu/data_source.py modified: plugins/send_setu_/update_setu/data_source.py modified: plugins/sign_in/goods_register.py modified: plugins/sign_in/group_user_checkin.py modified: plugins/sign_in/random_event.py modified: plugins/sign_in/utils.py modified: plugins/statistics/_model.py modified: plugins/statistics/statistics_handle.py modified: plugins/statistics/statistics_hook.py modified: plugins/update_picture.py modified: plugins/web_ui/api/request.py modified: plugins/word_bank/_model.py deleted: plugins/word_bank/_old_model.py modified: plugins/word_bank/_rule.py modified: plugins/word_bank/word_handle.py modified: plugins/word_clouds/data_source.py modified: resources/image/sign/sign_res/bar.png modified: resources/image/sign/sign_res/bar_white.png modified: services/db_context.py modified: services/log.py modified: utils/browser.py modified: utils/data_utils.py modified: utils/depends/__init__.py modified: utils/http_utils.py modified: utils/image_utils.py modified: utils/manager/admin_manager.py modified: utils/message_builder.py modified: utils/utils.py
286 lines
11 KiB
Python
Executable File
286 lines
11 KiB
Python
Executable File
from typing import Any, Optional, Tuple
|
||
|
||
import nonebot
|
||
from nonebot import Driver, on_command, on_regex
|
||
from nonebot.adapters.onebot.v11 import Bot, GroupMessageEvent, Message, MessageEvent
|
||
from nonebot.params import ArgStr, CommandArg, RegexGroup
|
||
from nonebot.typing import T_State
|
||
|
||
from configs.config import Config
|
||
from models.level_user import LevelUser
|
||
from services.log import logger
|
||
from utils.image_utils import text2image
|
||
from utils.manager import group_manager
|
||
from utils.message_builder import image
|
||
from utils.utils import get_bot, is_number, scheduler
|
||
|
||
from .data_source import (
|
||
BilibiliSub,
|
||
SubManager,
|
||
add_live_sub,
|
||
add_season_sub,
|
||
add_up_sub,
|
||
delete_sub,
|
||
get_media_id,
|
||
get_sub_status,
|
||
)
|
||
|
||
__zx_plugin_name__ = "B站订阅"
|
||
__plugin_usage__ = """
|
||
usage:
|
||
B站直播,番剧,UP动态开播等提醒
|
||
主播订阅相当于 直播间订阅 + UP订阅
|
||
指令:[示例Id乱打的,仅做示例]
|
||
添加订阅 ['主播'/'UP'/'番剧'] [id/链接/番名]
|
||
删除订阅 ['主播'/'UP'/'id'] [id]
|
||
查看订阅
|
||
示例:添加订阅主播 2345344 <-(直播房间id)
|
||
示例:添加订阅UP 2355543 <-(个人主页id)
|
||
示例:添加订阅番剧 史莱姆 <-(支持模糊搜索)
|
||
示例:添加订阅番剧 125344 <-(番剧id)
|
||
示例:删除订阅id 2324344 <-(任意id,通过查看订阅获取)
|
||
""".strip()
|
||
__plugin_des__ = "非常便利的B站订阅通知"
|
||
__plugin_cmd__ = ["添加订阅 [主播/UP/番剧] [id/链接/番名]", "删除订阅 ['主播'/'UP'/'id'] [id]", "查看订阅"]
|
||
__plugin_version__ = 0.1
|
||
__plugin_author__ = "HibiKier & NumberSir"
|
||
__plugin_settings__ = {
|
||
"level": 5,
|
||
"default_status": True,
|
||
"limit_superuser": False,
|
||
"cmd": ["B站订阅", "b站订阅", "添加订阅", "删除订阅", "查看订阅"],
|
||
}
|
||
__plugin_configs__ = {
|
||
"GROUP_BILIBILI_SUB_LEVEL": {
|
||
"value": 5,
|
||
"help": "群内bilibili订阅需要管理的权限",
|
||
"default_value": 5,
|
||
},
|
||
"LIVE_MSG_AT_ALL": {
|
||
"value": False,
|
||
"help": "直播提醒是否AT全体(仅在真寻是管理员时生效)",
|
||
"default_value": False,
|
||
},
|
||
"UP_MSG_AT_ALL": {
|
||
"value": False,
|
||
"help": "UP动态投稿提醒是否AT全体(仅在真寻是管理员时生效)",
|
||
"default_value": False,
|
||
},
|
||
}
|
||
|
||
add_sub = on_command("添加订阅", priority=5, block=True)
|
||
del_sub = on_command("删除订阅", priority=5, block=True)
|
||
show_sub_info = on_regex("^查看订阅$", priority=5, block=True)
|
||
|
||
driver: Driver = nonebot.get_driver()
|
||
|
||
|
||
sub_manager: Optional[SubManager] = None
|
||
|
||
|
||
@driver.on_startup
|
||
async def _():
|
||
global sub_manager
|
||
sub_manager = SubManager()
|
||
|
||
|
||
@add_sub.handle()
|
||
@del_sub.handle()
|
||
async def _(event: MessageEvent, state: T_State, arg: Message = CommandArg()):
|
||
msg = arg.extract_plain_text().strip().split()
|
||
if len(msg) < 2:
|
||
await add_sub.finish("参数不完全,请查看订阅帮助...")
|
||
sub_type = msg[0]
|
||
id_ = ""
|
||
if isinstance(event, GroupMessageEvent):
|
||
if not await LevelUser.check_level(
|
||
event.user_id,
|
||
event.group_id,
|
||
Config.get_config("bilibili_sub", "GROUP_BILIBILI_SUB_LEVEL"),
|
||
):
|
||
await add_sub.finish(
|
||
f"您的权限不足,群内订阅的需要 {Config.get_config('bilibili_sub', 'GROUP_BILIBILI_SUB_LEVEL')} 级权限..",
|
||
at_sender=True,
|
||
)
|
||
sub_user = f"{event.user_id}:{event.group_id}"
|
||
else:
|
||
sub_user = f"{event.user_id}"
|
||
state["sub_type"] = sub_type
|
||
state["sub_user"] = sub_user
|
||
if len(msg) > 1:
|
||
if "http" in msg[1]:
|
||
msg[1] = msg[1].split("?")[0]
|
||
msg[1] = msg[1][:-1] if msg[1][-1] == "/" else msg[1]
|
||
msg[1] = msg[1].split("/")[-1]
|
||
id_ = msg[1][2:] if msg[1].startswith("md") else msg[1]
|
||
if not is_number(id_):
|
||
if sub_type in ["season", "动漫", "番剧"]:
|
||
rst = "*以为您找到以下番剧,请输入Id选择:*\n"
|
||
state["season_data"] = await get_media_id(id_)
|
||
if len(state["season_data"]) == 0:
|
||
await add_sub.finish(f"未找到番剧:{msg}")
|
||
for i, x in enumerate(state["season_data"]):
|
||
rst += f'{i + 1}.{state["season_data"][x]["title"]}\n----------\n'
|
||
await add_sub.send("\n".join(rst.split("\n")[:-1]))
|
||
else:
|
||
await add_sub.finish("Id 必须为全数字!")
|
||
else:
|
||
state["id"] = int(id_)
|
||
|
||
|
||
@add_sub.got("sub_type")
|
||
@add_sub.got("sub_user")
|
||
@add_sub.got("id")
|
||
async def _(
|
||
event: MessageEvent,
|
||
state: T_State,
|
||
id_: str = ArgStr("id"),
|
||
sub_type: str = ArgStr("sub_type"),
|
||
sub_user: str = ArgStr("sub_user"),
|
||
):
|
||
if sub_type in ["season", "动漫", "番剧"] and state.get("season_data"):
|
||
season_data = state["season_data"]
|
||
if not is_number(id_) or int(id_) < 1 or int(id_) > len(season_data):
|
||
await add_sub.reject_arg("id", "Id必须为数字且在范围内!请重新输入...")
|
||
id_ = season_data[int(id_) - 1]["media_id"]
|
||
id_ = int(id_)
|
||
if sub_type in ["主播", "直播"]:
|
||
await add_sub.send(await add_live_sub(id_, sub_user))
|
||
elif sub_type.lower() in ["up", "用户"]:
|
||
await add_sub.send(await add_up_sub(id_, sub_user))
|
||
elif sub_type in ["season", "动漫", "番剧"]:
|
||
await add_sub.send(await add_season_sub(id_, sub_user))
|
||
else:
|
||
await add_sub.finish("参数错误,第一参数必须为:主播/up/番剧!")
|
||
logger.info(
|
||
f"(USER {event.user_id}, GROUP "
|
||
f"{event.group_id if isinstance(event, GroupMessageEvent) else 'private'})"
|
||
f" 添加订阅:{sub_type} -> {sub_user} -> {id_}"
|
||
)
|
||
|
||
|
||
@del_sub.got("sub_type")
|
||
@del_sub.got("sub_user")
|
||
@del_sub.got("id")
|
||
async def _(
|
||
event: MessageEvent,
|
||
id_: str = ArgStr("id"),
|
||
sub_type: str = ArgStr("sub_type"),
|
||
sub_user: str = ArgStr("sub_user"),
|
||
):
|
||
if sub_type in ["主播", "直播"]:
|
||
result = await BilibiliSub.delete_bilibili_sub(int(id_), sub_user, "live")
|
||
elif sub_type.lower() in ["up", "用户"]:
|
||
result = await BilibiliSub.delete_bilibili_sub(int(id_), sub_user, "up")
|
||
else:
|
||
result = await BilibiliSub.delete_bilibili_sub(int(id_), sub_user)
|
||
if result:
|
||
await del_sub.send(f"删除订阅id:{id_} 成功...")
|
||
logger.info(
|
||
f"(USER {event.user_id}, GROUP "
|
||
f"{event.group_id if isinstance(event, GroupMessageEvent) else 'private'})"
|
||
f" 删除订阅 {id_}"
|
||
)
|
||
else:
|
||
await del_sub.send(f"删除订阅id:{id_} 失败...")
|
||
|
||
|
||
@show_sub_info.handle()
|
||
async def _(event: MessageEvent):
|
||
if isinstance(event, GroupMessageEvent):
|
||
id_ = f"{event.group_id}"
|
||
else:
|
||
id_ = f"{event.user_id}"
|
||
data = await BilibiliSub.filter(sub_users__contains=id_).all()
|
||
live_rst = ""
|
||
up_rst = ""
|
||
season_rst = ""
|
||
for x in data:
|
||
if x.sub_type == "live":
|
||
live_rst += (
|
||
f"\t直播间id:{x.sub_id}\n" f"\t名称:{x.uname}\n" f"------------------\n"
|
||
)
|
||
if x.sub_type == "up":
|
||
up_rst += f"\tUP:{x.uname}\n" f"\tuid:{x.uid}\n" f"------------------\n"
|
||
if x.sub_type == "season":
|
||
season_rst += (
|
||
f"\t番剧id:{x.sub_id}\n"
|
||
f"\t番名:{x.season_name}\n"
|
||
f"\t当前集数:{x.season_current_episode}\n"
|
||
f"------------------\n"
|
||
)
|
||
live_rst = "当前订阅的直播:\n" + live_rst if live_rst else live_rst
|
||
up_rst = "当前订阅的UP:\n" + up_rst if up_rst else up_rst
|
||
season_rst = "当前订阅的番剧:\n" + season_rst if season_rst else season_rst
|
||
if not live_rst and not up_rst and not season_rst:
|
||
live_rst = (
|
||
"该群目前没有任何订阅..." if isinstance(event, GroupMessageEvent) else "您目前没有任何订阅..."
|
||
)
|
||
await show_sub_info.send(
|
||
image(
|
||
await text2image(
|
||
live_rst + up_rst + season_rst, padding=10, color="#f9f6f2"
|
||
)
|
||
)
|
||
)
|
||
|
||
|
||
# 推送
|
||
@scheduler.scheduled_job(
|
||
"interval",
|
||
seconds=30,
|
||
)
|
||
async def _():
|
||
bot = get_bot()
|
||
sub = None
|
||
if bot:
|
||
# try:
|
||
await sub_manager.reload_sub_data()
|
||
sub = await sub_manager.random_sub_data()
|
||
if sub:
|
||
logger.debug(f"Bilibili订阅开始检测:{sub.sub_id}")
|
||
rst = await get_sub_status(sub.sub_id, sub.sub_type)
|
||
await send_sub_msg(rst, sub, bot)
|
||
if sub.sub_type == "live":
|
||
rst = await get_sub_status(sub.sub_id, "up")
|
||
await send_sub_msg(rst, sub, bot)
|
||
# except Exception as e:
|
||
# logger.error(f"B站订阅推送发生错误 sub_id:{sub.sub_id if sub else 0} {type(e)}:{e}")
|
||
|
||
|
||
async def send_sub_msg(rst: str, sub: BilibiliSub, bot: Bot):
|
||
"""
|
||
推送信息
|
||
:param rst: 回复
|
||
:param sub: BilibiliSub
|
||
:param bot: Bot
|
||
"""
|
||
temp_group = []
|
||
if rst:
|
||
for x in sub.sub_users.split(",")[:-1]:
|
||
try:
|
||
if ":" in x and x.split(":")[1] not in temp_group:
|
||
group_id = int(x.split(":")[1])
|
||
temp_group.append(group_id)
|
||
if (
|
||
await bot.get_group_member_info(
|
||
group_id=group_id, user_id=int(bot.self_id), no_cache=True
|
||
)
|
||
)["role"] in ["owner", "admin"]:
|
||
if (
|
||
sub.sub_type == "live"
|
||
and Config.get_config("bilibili_sub", "LIVE_MSG_AT_ALL")
|
||
) or (
|
||
sub.sub_type == "up"
|
||
and Config.get_config("bilibili_sub", "UP_MSG_AT_ALL")
|
||
):
|
||
rst = "[CQ:at,qq=all]\n" + rst
|
||
if group_manager.get_plugin_status("bilibili_sub", group_id):
|
||
await bot.send_group_msg(
|
||
group_id=group_id, message=Message(rst)
|
||
)
|
||
else:
|
||
await bot.send_private_msg(user_id=int(x), message=Message(rst))
|
||
except Exception as e:
|
||
logger.error(f"B站订阅推送发生错误 sub_id:{sub.sub_id} {type(e)}:{e}")
|