from typing import Optional, List, Dict, Tuple, TYPE_CHECKING from simnet.errors import BadRequest as SimnetBadRequest from simnet.models.starrail.character import RogueCharacter from simnet.models.starrail.chronicle.rogue import StarRailRogue from telegram import Update, Message from telegram.constants import ChatAction from telegram.ext import CallbackContext, filters 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, CookiesNotFoundError from utils.log import logger from utils.uid import mask_number if TYPE_CHECKING: from simnet import StarRailClient __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) try: async with self.helper.genshin(user.id) as client: if client.player_id != uid: raise CookiesNotFoundError(uid) render_result = await self.render(client, pre, uid) except CookiesNotFoundError: async with self.helper.public_genshin(user.id) as client: render_result = await self.render(client, pre, uid) except SimnetBadRequest 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"{user.id}.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": mask_number(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: "StarRailClient", pre: bool, uid: Optional[int] = None) -> RenderResult: if uid is None: uid = client.player_id 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", )