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
385 lines
14 KiB
Python
Executable File
385 lines
14 KiB
Python
Executable File
import random
|
||
from asyncio.exceptions import TimeoutError
|
||
from datetime import datetime
|
||
from typing import Optional, Tuple
|
||
|
||
# from .utils import get_videos
|
||
from bilireq import dynamic
|
||
from bilireq.exceptions import ResponseCodeError
|
||
from bilireq.live import get_room_info_by_id
|
||
from bilireq.user import get_videos
|
||
from nonebot.adapters.onebot.v11 import MessageSegment
|
||
|
||
from configs.path_config import IMAGE_PATH, TEMP_PATH
|
||
from services.log import logger
|
||
from utils.browser import get_browser
|
||
from utils.http_utils import AsyncHttpx, AsyncPlaywright
|
||
from utils.manager import resources_manager
|
||
from utils.message_builder import image
|
||
from utils.utils import get_bot
|
||
|
||
from .model import BilibiliSub
|
||
from .utils import get_meta, get_user_card
|
||
|
||
SEARCH_URL = "https://api.bilibili.com/x/web-interface/search/all/v2"
|
||
|
||
DYNAMIC_PATH = IMAGE_PATH / "bilibili_sub" / "dynamic"
|
||
DYNAMIC_PATH.mkdir(exist_ok=True, parents=True)
|
||
|
||
|
||
resources_manager.add_temp_dir(DYNAMIC_PATH)
|
||
|
||
|
||
async def add_live_sub(live_id: int, sub_user: str) -> str:
|
||
"""
|
||
添加直播订阅
|
||
:param live_id: 直播房间号
|
||
:param sub_user: 订阅用户 id # 7384933:private or 7384933:2342344(group)
|
||
:return:
|
||
"""
|
||
try:
|
||
try:
|
||
"""bilibili_api.live库的LiveRoom类中get_room_info改为bilireq.live库的get_room_info_by_id方法"""
|
||
live_info = await get_room_info_by_id(live_id)
|
||
except ResponseCodeError:
|
||
return f"未找到房间号Id:{live_id} 的信息,请检查Id是否正确"
|
||
uid = live_info["uid"]
|
||
room_id = live_info["room_id"]
|
||
short_id = live_info["short_id"]
|
||
title = live_info["title"]
|
||
live_status = live_info["live_status"]
|
||
if await BilibiliSub.sub_handle(
|
||
room_id,
|
||
"live",
|
||
sub_user,
|
||
uid=uid,
|
||
live_short_id=short_id,
|
||
live_status=live_status,
|
||
):
|
||
await _get_up_status(room_id)
|
||
uname = (await BilibiliSub.get_or_none(sub_id=room_id)).uname
|
||
return (
|
||
"已成功订阅主播:\n"
|
||
f"\ttitle:{title}\n"
|
||
f"\tname: {uname}\n"
|
||
f"\tlive_id:{room_id}\n"
|
||
f"\tuid:{uid}"
|
||
)
|
||
else:
|
||
return "添加订阅失败..."
|
||
except Exception as e:
|
||
logger.error(f"订阅主播live_id:{live_id} 发生了错误 {type(e)}:{e}")
|
||
return "添加订阅失败..."
|
||
|
||
|
||
async def add_up_sub(uid: int, sub_user: str) -> str:
|
||
"""
|
||
添加订阅 UP
|
||
:param uid: UP uid
|
||
:param sub_user: 订阅用户
|
||
"""
|
||
try:
|
||
try:
|
||
"""bilibili_api.user库中User类的get_user_info改为bilireq.user库的get_user_info方法"""
|
||
user_info = await get_user_card(uid)
|
||
except ResponseCodeError:
|
||
return f"未找到UpId:{uid} 的信息,请检查Id是否正确"
|
||
uname = user_info["name"]
|
||
"""bilibili_api.user库中User类的get_dynamics改为bilireq.dynamic库的get_user_dynamics方法"""
|
||
dynamic_info = await dynamic.get_user_dynamics(uid)
|
||
dynamic_upload_time = 0
|
||
if dynamic_info.get("cards"):
|
||
dynamic_upload_time = dynamic_info["cards"][0]["desc"]["timestamp"]
|
||
"""bilibili_api.user库中User类的get_videos改为bilireq.user库的get_videos方法"""
|
||
video_info = await get_videos(uid)
|
||
latest_video_created = 0
|
||
if video_info["list"].get("vlist"):
|
||
latest_video_created = video_info["list"]["vlist"][0]["created"]
|
||
if await BilibiliSub.sub_handle(
|
||
uid,
|
||
"up",
|
||
sub_user,
|
||
uid=uid,
|
||
uname=uname,
|
||
dynamic_upload_time=dynamic_upload_time,
|
||
latest_video_created=latest_video_created,
|
||
):
|
||
return "已成功订阅UP:\n" f"\tname: {uname}\n" f"\tuid:{uid}"
|
||
else:
|
||
return "添加订阅失败..."
|
||
except Exception as e:
|
||
logger.error(f"订阅Up uid:{uid} 发生了错误 {type(e)}:{e}")
|
||
return "添加订阅失败..."
|
||
|
||
|
||
async def add_season_sub(media_id: int, sub_user: str) -> str:
|
||
"""
|
||
添加订阅 UP
|
||
:param media_id: 番剧 media_id
|
||
:param sub_user: 订阅用户
|
||
"""
|
||
try:
|
||
try:
|
||
"""bilibili_api.bangumi库中get_meta改为bilireq.bangumi库的get_meta方法"""
|
||
season_info = await get_meta(media_id)
|
||
except ResponseCodeError:
|
||
return f"未找到media_id:{media_id} 的信息,请检查Id是否正确"
|
||
season_id = season_info["media"]["season_id"]
|
||
season_current_episode = season_info["media"]["new_ep"]["index"]
|
||
season_name = season_info["media"]["title"]
|
||
if await BilibiliSub.sub_handle(
|
||
media_id,
|
||
"season",
|
||
sub_user,
|
||
season_name=season_name,
|
||
season_id=season_id,
|
||
season_current_episode=season_current_episode,
|
||
):
|
||
return (
|
||
"已成功订阅番剧:\n"
|
||
f"\ttitle: {season_name}\n"
|
||
f"\tcurrent_episode: {season_current_episode}"
|
||
)
|
||
else:
|
||
return "添加订阅失败..."
|
||
except Exception as e:
|
||
logger.error(f"订阅番剧 media_id:{media_id} 发生了错误 {type(e)}:{e}")
|
||
return "添加订阅失败..."
|
||
|
||
|
||
async def delete_sub(sub_id: str, sub_user: str) -> str:
|
||
"""
|
||
删除订阅
|
||
:param sub_id: 订阅 id
|
||
:param sub_user: 订阅用户 id # 7384933:private or 7384933:2342344(group)
|
||
"""
|
||
if await BilibiliSub.delete_bilibili_sub(int(sub_id), sub_user):
|
||
return f"已成功取消订阅:{sub_id}"
|
||
else:
|
||
return f"取消订阅:{sub_id} 失败,请检查是否订阅过该Id...."
|
||
|
||
|
||
async def get_media_id(keyword: str) -> dict:
|
||
"""
|
||
获取番剧的 media_id
|
||
:param keyword: 番剧名称
|
||
"""
|
||
params = {"keyword": keyword}
|
||
for _ in range(3):
|
||
try:
|
||
_season_data = {}
|
||
response = await AsyncHttpx.get(SEARCH_URL, params=params, timeout=5)
|
||
if response.status_code == 200:
|
||
data = response.json()
|
||
if data.get("data"):
|
||
for item in data["data"]["result"]:
|
||
if item["result_type"] == "media_bangumi":
|
||
idx = 0
|
||
for x in item["data"]:
|
||
_season_data[idx] = {
|
||
"media_id": x["media_id"],
|
||
"title": x["title"]
|
||
.replace('<em class="keyword">', "")
|
||
.replace("</em>", ""),
|
||
}
|
||
idx += 1
|
||
return _season_data
|
||
except TimeoutError:
|
||
pass
|
||
return {}
|
||
|
||
|
||
async def get_sub_status(id_: int, sub_type: str) -> Optional[str]:
|
||
"""
|
||
获取订阅状态
|
||
:param id_: 订阅 id
|
||
:param sub_type: 订阅类型
|
||
"""
|
||
try:
|
||
if sub_type == "live":
|
||
return await _get_live_status(id_)
|
||
elif sub_type == "up":
|
||
return await _get_up_status(id_)
|
||
elif sub_type == "season":
|
||
return await _get_season_status(id_)
|
||
except ResponseCodeError as msg:
|
||
logger.info(f"Id:{id_} 获取信息失败...{msg}")
|
||
return None
|
||
# return f"Id:{id_} 获取信息失败...请检查订阅Id是否存在或稍后再试..."
|
||
# except Exception as e:
|
||
# logger.error(f"获取订阅状态发生预料之外的错误 id_:{id_} {type(e)}:{e}")
|
||
# return "发生了预料之外的错误..请稍后再试或联系管理员....."
|
||
|
||
|
||
async def _get_live_status(id_: int) -> Optional[str]:
|
||
"""
|
||
获取直播订阅状态
|
||
:param id_: 直播间 id
|
||
"""
|
||
"""bilibili_api.live库的LiveRoom类中get_room_info改为bilireq.live库的get_room_info_by_id方法"""
|
||
live_info = await get_room_info_by_id(id_)
|
||
title = live_info["title"]
|
||
room_id = live_info["room_id"]
|
||
live_status = live_info["live_status"]
|
||
cover = live_info["user_cover"]
|
||
sub = await BilibiliSub.get_or_none(sub_id=id_)
|
||
if sub.live_status != live_status:
|
||
await BilibiliSub.sub_handle(id_, live_status=live_status)
|
||
if sub.live_status in [0, 2] and live_status == 1:
|
||
return (
|
||
f""
|
||
f"{image(cover)}\n"
|
||
f"{sub.uname} 开播啦!\n"
|
||
f"标题:{title}\n"
|
||
f"直链:https://live.bilibili.com/{room_id}"
|
||
)
|
||
return None
|
||
|
||
|
||
async def _get_up_status(id_: int) -> Optional[str]:
|
||
"""
|
||
获取用户投稿状态
|
||
:param id_: 订阅 id
|
||
:return:
|
||
"""
|
||
_user = await BilibiliSub.get_or_none(sub_id=id_)
|
||
"""bilibili_api.user库中User类的get_user_info改为bilireq.user库的get_user_info方法"""
|
||
user_info = await get_user_card(_user.uid)
|
||
uname = user_info["name"]
|
||
"""bilibili_api.user库中User类的get_videos改为bilireq.user库的get_videos方法"""
|
||
video_info = await get_videos(_user.uid)
|
||
latest_video_created = 0
|
||
video = None
|
||
dividing_line = "\n-------------\n"
|
||
if _user.uname != uname:
|
||
await BilibiliSub.sub_handle(id_, uname=uname)
|
||
dynamic_img, dynamic_upload_time, link = await get_user_dynamic(_user.uid, _user)
|
||
if video_info["list"].get("vlist"):
|
||
video = video_info["list"]["vlist"][0]
|
||
latest_video_created = video["created"]
|
||
rst = ""
|
||
if dynamic_img:
|
||
await BilibiliSub.sub_handle(id_, dynamic_upload_time=dynamic_upload_time)
|
||
rst += f"{uname} 发布了动态!\n" f"{dynamic_img}\n{link}"
|
||
if (
|
||
latest_video_created
|
||
and _user.latest_video_created
|
||
and video
|
||
and _user.latest_video_created < latest_video_created
|
||
):
|
||
rst = rst + dividing_line if rst else rst
|
||
await BilibiliSub.sub_handle(id_, latest_video_created=latest_video_created)
|
||
rst += (
|
||
f'{image(video["pic"])}\n'
|
||
f"{uname} 投稿了新视频啦\n"
|
||
f'标题:{video["title"]}\n'
|
||
f'Bvid:{video["bvid"]}\n'
|
||
f'直链:https://www.bilibili.com/video/{video["bvid"]}'
|
||
)
|
||
rst = None if rst == dividing_line else rst
|
||
return rst
|
||
|
||
|
||
async def _get_season_status(id_) -> Optional[str]:
|
||
"""
|
||
获取 番剧 更新状态
|
||
:param id_: 番剧 id
|
||
"""
|
||
"""bilibili_api.bangumi库中get_meta改为bilireq.bangumi库的get_meta方法"""
|
||
season_info = await get_meta(id_)
|
||
title = season_info["media"]["title"]
|
||
_idx = (await BilibiliSub.get_or_none(sub_id=id_)).season_current_episode
|
||
new_ep = season_info["media"]["new_ep"]["index"]
|
||
if new_ep != _idx:
|
||
await BilibiliSub.sub_handle(
|
||
id_, season_current_episode=new_ep, season_update_time=datetime.now()
|
||
)
|
||
return (
|
||
f'{image(season_info["media"]["cover"])}\n'
|
||
f"[{title}]更新啦\n"
|
||
f"最新集数:{new_ep}"
|
||
)
|
||
return None
|
||
|
||
|
||
async def get_user_dynamic(
|
||
uid: int, local_user: BilibiliSub
|
||
) -> Tuple[Optional[MessageSegment], int, str]:
|
||
"""
|
||
获取用户动态
|
||
:param uid: 用户uid
|
||
:param local_user: 数据库存储的用户数据
|
||
:return: 最新动态截图与时间
|
||
"""
|
||
"""bilibili_api.user库中User类的get_dynamics改为bilireq.dynamic库的get_user_dynamics方法"""
|
||
dynamic_info = await dynamic.get_user_dynamics(uid)
|
||
if dynamic_info.get("cards"):
|
||
dynamic_upload_time = dynamic_info["cards"][0]["desc"]["timestamp"]
|
||
dynamic_id = dynamic_info["cards"][0]["desc"]["dynamic_id"]
|
||
if local_user.dynamic_upload_time < dynamic_upload_time:
|
||
image = await AsyncPlaywright.screenshot(
|
||
f"https://t.bilibili.com/{dynamic_id}",
|
||
DYNAMIC_PATH / f"sub_{local_user.sub_id}.png",
|
||
".bili-dyn-item__main",
|
||
wait_until="networkidle",
|
||
)
|
||
return (
|
||
image,
|
||
dynamic_upload_time,
|
||
f"https://t.bilibili.com/{dynamic_id}",
|
||
)
|
||
return None, 0, ""
|
||
|
||
|
||
class SubManager:
|
||
def __init__(self):
|
||
self.live_data = []
|
||
self.up_data = []
|
||
self.season_data = []
|
||
self.current_index = -1
|
||
|
||
async def reload_sub_data(self):
|
||
"""
|
||
重载数据
|
||
"""
|
||
if not self.live_data or not self.up_data or not self.season_data:
|
||
(
|
||
_live_data,
|
||
_up_data,
|
||
_season_data,
|
||
) = await BilibiliSub.get_all_sub_data()
|
||
if not self.live_data:
|
||
self.live_data = _live_data
|
||
if not self.up_data:
|
||
self.up_data = _up_data
|
||
if not self.season_data:
|
||
self.season_data = _season_data
|
||
|
||
async def random_sub_data(self) -> Optional[BilibiliSub]:
|
||
"""
|
||
随机获取一条数据
|
||
:return:
|
||
"""
|
||
sub = None
|
||
if not self.live_data and not self.up_data and not self.season_data:
|
||
return sub
|
||
self.current_index += 1
|
||
if self.current_index == 0:
|
||
if self.live_data:
|
||
sub = random.choice(self.live_data)
|
||
self.live_data.remove(sub)
|
||
elif self.current_index == 1:
|
||
if self.up_data:
|
||
sub = random.choice(self.up_data)
|
||
self.up_data.remove(sub)
|
||
elif self.current_index == 2:
|
||
if self.season_data:
|
||
sub = random.choice(self.season_data)
|
||
self.season_data.remove(sub)
|
||
else:
|
||
self.current_index = -1
|
||
if sub:
|
||
return sub
|
||
await self.reload_sub_data()
|
||
return await self.random_sub_data()
|