mirror of
https://github.com/PaiGramTeam/PamGram.git
synced 2024-11-29 00:42:44 +00:00
175 lines
7.3 KiB
Python
175 lines
7.3 KiB
Python
from typing import Optional, List, Dict, Tuple
|
||
|
||
from genshin import Client, GenshinException
|
||
from genshin.models import StarRailRogue, RogueCharacter
|
||
from telegram import Update, InlineKeyboardButton, InlineKeyboardMarkup, Message
|
||
from telegram.constants import ChatAction
|
||
from telegram.ext import CallbackContext, filters
|
||
from telegram.helpers import create_deep_linked_url
|
||
|
||
from core.dependence.assets import AssetsService
|
||
from core.plugin import Plugin, handler
|
||
from core.services.cookies.error import TooManyRequestPublicCookies
|
||
from core.services.template.models import RenderResult
|
||
from core.services.template.services import TemplateService
|
||
from plugins.tools.genshin import GenshinHelper, PlayerNotFoundError, CookiesNotFoundError
|
||
from utils.log import logger
|
||
|
||
__all__ = ("PlayerRoguePlugins",)
|
||
|
||
|
||
class NotSupport(Exception):
|
||
"""不支持的服务器"""
|
||
|
||
|
||
class NotHaveData(Exception):
|
||
"""没有数据"""
|
||
|
||
|
||
class PlayerRoguePlugins(Plugin):
|
||
"""玩家模拟宇宙信息查询"""
|
||
|
||
def __init__(
|
||
self,
|
||
template: TemplateService,
|
||
assets: AssetsService,
|
||
helper: GenshinHelper,
|
||
):
|
||
self.template_service = template
|
||
self.assets = assets
|
||
self.helper = helper
|
||
|
||
async def get_uid(self, user_id: int, args: List[str], reply: Optional[Message]) -> Tuple[int, bool]:
|
||
"""通过消息获取 uid,优先级:args > reply > self"""
|
||
uid, user_id_, pre = None, user_id, False
|
||
if args:
|
||
for i in args:
|
||
if i is not None:
|
||
if i.isdigit() and len(i) == 9:
|
||
uid = int(i)
|
||
if "上" in i:
|
||
pre = True
|
||
if reply:
|
||
try:
|
||
user_id_ = reply.from_user.id
|
||
except AttributeError:
|
||
pass
|
||
if not uid:
|
||
player_info = await self.helper.players_service.get_player(user_id_)
|
||
if player_info is not None:
|
||
uid = player_info.player_id
|
||
if (not uid) and (user_id_ != user_id):
|
||
player_info = await self.helper.players_service.get_player(user_id)
|
||
if player_info is not None:
|
||
uid = player_info.player_id
|
||
return uid, pre
|
||
|
||
@handler.command("rogue", block=False)
|
||
@handler.message(filters.Regex("^模拟宇宙信息查询(.*)"), block=False)
|
||
async def command_start(self, update: Update, context: CallbackContext) -> Optional[int]:
|
||
user = update.effective_user
|
||
message = update.effective_message
|
||
logger.info("用户 %s[%s] 查询模拟宇宙信息命令请求", user.full_name, user.id)
|
||
try:
|
||
uid, pre = await self.get_uid(user.id, context.args, message.reply_to_message)
|
||
if uid and str(uid)[0] not in ["1", "2", "5"]:
|
||
# todo: 支持国际服
|
||
raise NotSupport
|
||
try:
|
||
client = await self.helper.get_genshin_client(user.id)
|
||
if client.uid != uid:
|
||
raise CookiesNotFoundError(uid)
|
||
except CookiesNotFoundError:
|
||
client, _ = await self.helper.get_public_genshin_client(user.id)
|
||
render_result = await self.render(client, pre, uid)
|
||
except PlayerNotFoundError:
|
||
buttons = [[InlineKeyboardButton("点我绑定账号", url=create_deep_linked_url(context.bot.username, "set_cookie"))]]
|
||
if filters.ChatType.GROUPS.filter(message):
|
||
reply_message = await message.reply_text(
|
||
"未查询到您所绑定的账号信息,请先私聊彦卿绑定账号", reply_markup=InlineKeyboardMarkup(buttons)
|
||
)
|
||
self.add_delete_message_job(reply_message, delay=30)
|
||
self.add_delete_message_job(message, delay=30)
|
||
else:
|
||
await message.reply_text("未查询到您所绑定的账号信息,请先绑定账号", reply_markup=InlineKeyboardMarkup(buttons))
|
||
return
|
||
except GenshinException as exc:
|
||
if exc.retcode == 1034:
|
||
await message.reply_text("出错了呜呜呜 ~ 请稍后重试")
|
||
return
|
||
raise exc
|
||
except TooManyRequestPublicCookies:
|
||
await message.reply_text("用户查询次数过多 请稍后重试")
|
||
return
|
||
except AttributeError as exc:
|
||
logger.error("模拟宇宙数据有误")
|
||
logger.exception(exc)
|
||
await message.reply_text("模拟宇宙数据有误 估计是彦卿晕了")
|
||
return
|
||
except NotSupport:
|
||
reply_message = await message.reply_text("暂不支持该服务器查询模拟宇宙数据")
|
||
if filters.ChatType.GROUPS.filter(reply_message):
|
||
self.add_delete_message_job(message)
|
||
self.add_delete_message_job(reply_message)
|
||
return
|
||
except NotHaveData:
|
||
reply_message = await message.reply_text("没有查找到模拟宇宙数据")
|
||
if filters.ChatType.GROUPS.filter(reply_message):
|
||
self.add_delete_message_job(message)
|
||
self.add_delete_message_job(reply_message)
|
||
return
|
||
await message.reply_chat_action(ChatAction.UPLOAD_PHOTO)
|
||
await render_result.reply_photo(message, filename=f"{client.uid}.png", allow_sending_without_reply=True)
|
||
|
||
async def get_rander_data(self, uid: int, data: StarRailRogue, pre: bool) -> Dict:
|
||
luo_ma_bum = ["", "Ⅰ", "Ⅱ", "Ⅲ", "Ⅳ", "Ⅴ", "Ⅵ"]
|
||
buff_en_map = {
|
||
"「丰饶」": "Abundance",
|
||
"「毁灭」": "Destruction",
|
||
"「智识」": "Erudition",
|
||
"「协同」": "Harmony",
|
||
"「巡猎」": "Hunt",
|
||
"「欢愉」": "Joy",
|
||
"「记忆」": "Memory",
|
||
"「虚无」": "Nihility",
|
||
"「存护」": "Preservation",
|
||
}
|
||
record_raw = data.last_record if pre else data.current_record
|
||
if not record_raw.has_data:
|
||
raise NotHaveData
|
||
record = record_raw.records[0]
|
||
avatars = record.final_lineup
|
||
new_avatars = [None, None, None, None]
|
||
for idx, avatar in enumerate(avatars):
|
||
old_avatar = avatar.dict()
|
||
old_avatar["icon"] = self.assets.avatar.square(avatar.id).as_uri()
|
||
new_avatars[idx] = RogueCharacter(**old_avatar)
|
||
|
||
return {
|
||
"uid": uid,
|
||
"basic": data.basic_info,
|
||
"name": f"{record.name} {luo_ma_bum[record.difficulty]}",
|
||
"finish_cnt": record_raw.basic.finish_cnt,
|
||
"time": record.finish_time.datetime.strftime("%Y-%m-%d %H:%M"),
|
||
"score": record.score,
|
||
"avatars": new_avatars,
|
||
"buffs": record.buffs,
|
||
"buff_en_map": buff_en_map,
|
||
"miracles": record.miracles,
|
||
}
|
||
|
||
async def render(self, client: Client, pre: bool, uid: Optional[int] = None) -> RenderResult:
|
||
if uid is None:
|
||
uid = client.uid
|
||
|
||
rogue = await client.get_starrail_rogue(uid)
|
||
data = await self.get_rander_data(uid, rogue, pre)
|
||
|
||
return await self.template_service.render(
|
||
"starrail/rogue/rogue.html",
|
||
data,
|
||
{"width": 520, "height": 1000},
|
||
full_page=True,
|
||
query_selector="#new-container",
|
||
)
|