2022-11-12 12:59:42 +00:00
|
|
|
|
from typing import Tuple, Optional
|
|
|
|
|
|
2022-11-12 15:08:25 +00:00
|
|
|
|
from genshin import Region, GenshinException
|
2022-11-12 12:59:42 +00:00
|
|
|
|
from telegram import Update, InlineKeyboardButton, InlineKeyboardMarkup
|
|
|
|
|
from telegram.ext import CallbackContext
|
|
|
|
|
|
|
|
|
|
from core.base.redisdb import RedisDB
|
|
|
|
|
from core.baseplugin import BasePlugin
|
|
|
|
|
from core.config import config
|
|
|
|
|
from core.cookies import CookiesService
|
2022-11-12 15:08:25 +00:00
|
|
|
|
from core.cookies.error import CookiesNotFoundError
|
2022-11-12 12:59:42 +00:00
|
|
|
|
from core.plugin import Plugin, handler
|
|
|
|
|
from core.user import UserService
|
2022-11-12 15:08:25 +00:00
|
|
|
|
from core.user.error import UserNotFoundError
|
2022-11-13 05:45:01 +00:00
|
|
|
|
from modules.apihelper.error import ResponseException, APIHelperException
|
2022-11-12 12:59:42 +00:00
|
|
|
|
from modules.apihelper.hyperion import Verification
|
|
|
|
|
from utils.decorators.error import error_callable
|
|
|
|
|
from utils.decorators.restricts import restricts
|
2022-11-12 15:08:25 +00:00
|
|
|
|
from utils.helpers import get_genshin_client
|
2022-11-13 05:45:01 +00:00
|
|
|
|
from utils.log import logger
|
2022-11-12 12:59:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class VerificationSystem:
|
|
|
|
|
def __init__(self, redis: RedisDB = None):
|
|
|
|
|
self.cache = redis.client
|
|
|
|
|
self.qname = "plugin:verification:"
|
|
|
|
|
|
|
|
|
|
async def get_challenge(self, uid: int) -> Tuple[Optional[str], Optional[str]]:
|
|
|
|
|
data = await self.cache.get(f"{self.qname}{uid}")
|
|
|
|
|
if not data:
|
|
|
|
|
return None, None
|
|
|
|
|
data = data.decode("utf-8").split("|")
|
|
|
|
|
return data[0], data[1]
|
|
|
|
|
|
|
|
|
|
async def set_challenge(self, uid: int, gt: str, challenge: str):
|
|
|
|
|
await self.cache.set(f"{self.qname}{uid}", f"{gt}|{challenge}")
|
|
|
|
|
await self.cache.expire(f"{self.qname}{uid}", 10 * 60)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class VerificationPlugins(Plugin, BasePlugin):
|
|
|
|
|
def __init__(self, user_service: UserService = None, cookies_service: CookiesService = None, redis: RedisDB = None):
|
|
|
|
|
self.cookies_service = cookies_service
|
|
|
|
|
self.user_service = user_service
|
|
|
|
|
self.system = VerificationSystem(redis)
|
|
|
|
|
|
|
|
|
|
@handler.command("verify", block=False)
|
|
|
|
|
@restricts(restricts_time=60)
|
|
|
|
|
@error_callable
|
|
|
|
|
async def verify(self, update: Update, context: CallbackContext) -> None:
|
|
|
|
|
user = update.effective_user
|
|
|
|
|
message = update.effective_message
|
2022-11-13 05:45:01 +00:00
|
|
|
|
logger.info(f"用户 %s[%s] 发出verify命令", user.full_name, user.id)
|
2022-11-12 15:08:25 +00:00
|
|
|
|
try:
|
|
|
|
|
client = await get_genshin_client(user.id)
|
|
|
|
|
if client.region != Region.CHINESE:
|
|
|
|
|
await message.reply_text("非法用户")
|
|
|
|
|
return
|
|
|
|
|
except UserNotFoundError:
|
|
|
|
|
await message.reply_text("用户未找到")
|
2022-11-12 12:59:42 +00:00
|
|
|
|
return
|
2022-11-12 15:08:25 +00:00
|
|
|
|
except CookiesNotFoundError:
|
|
|
|
|
await message.reply_text("检测到用户为UID绑定,无需认证")
|
|
|
|
|
return
|
|
|
|
|
verification = Verification(cookies=client.cookie_manager.cookies)
|
2022-11-12 12:59:42 +00:00
|
|
|
|
if context.args and len(context.args) > 0:
|
|
|
|
|
validate = context.args[0]
|
2022-11-12 15:08:25 +00:00
|
|
|
|
_, challenge = await self.system.get_challenge(client.uid)
|
2022-11-12 12:59:42 +00:00
|
|
|
|
if challenge:
|
2022-11-13 05:45:01 +00:00
|
|
|
|
try:
|
|
|
|
|
await verification.verify(challenge, validate)
|
|
|
|
|
logger.success(f"用户 %s[%s] 验证成功", user.full_name, user.id)
|
|
|
|
|
await message.reply_text("验证成功")
|
|
|
|
|
except ResponseException as exc:
|
|
|
|
|
logger.warning(f"用户 %s[%s] 验证失效 API返回 [%s]%s", user.full_name, user.id, exc.code, exc.message)
|
|
|
|
|
await message.reply_text(f"验证失败 错误信息为 [{exc.code}]{exc.message} 请稍后重试")
|
2022-11-12 12:59:42 +00:00
|
|
|
|
else:
|
2022-11-13 05:45:01 +00:00
|
|
|
|
logger.warning(f"用户 %s[%s] 验证失效 请求已经过期", user.full_name, user.id)
|
|
|
|
|
await message.reply_text("验证失效 请求已经过期 请稍后重试")
|
2022-11-12 12:59:42 +00:00
|
|
|
|
return
|
2022-11-12 15:08:25 +00:00
|
|
|
|
try:
|
|
|
|
|
await client.get_genshin_notes()
|
|
|
|
|
except GenshinException as exc:
|
|
|
|
|
if exc.retcode != 1034:
|
|
|
|
|
raise exc
|
|
|
|
|
else:
|
|
|
|
|
await message.reply_text("账户正常,无需认证")
|
|
|
|
|
return
|
2022-11-13 05:45:01 +00:00
|
|
|
|
try:
|
|
|
|
|
data = await verification.create()
|
|
|
|
|
logger.success(f"用户 %s[%s] 创建验证成功", user.full_name, user.id)
|
|
|
|
|
except ResponseException as exc:
|
|
|
|
|
logger.warning(f"用户 %s[%s] 创建验证失效 API返回 [%s]%s 请稍后重试", user.full_name, user.id, exc.code, exc.message)
|
|
|
|
|
await message.reply_text(f"创建验证失败 错误信息为 [{exc.code}]{exc.message} 请稍后重试")
|
|
|
|
|
return
|
2022-11-12 12:59:42 +00:00
|
|
|
|
challenge = data["challenge"]
|
|
|
|
|
gt = data["gt"]
|
2022-11-13 05:45:01 +00:00
|
|
|
|
try:
|
|
|
|
|
validate = await verification.ajax(referer="https://webstatic.mihoyo.com/", gt=gt, challenge=challenge)
|
|
|
|
|
if validate:
|
|
|
|
|
await verification.verify(challenge, validate)
|
|
|
|
|
logger.success(f"用户 %s[%s] 通过 ajax 验证", user.full_name, user.id)
|
|
|
|
|
await message.reply_text("验证成功")
|
|
|
|
|
return
|
|
|
|
|
except APIHelperException as exc:
|
|
|
|
|
logger.warning(f"用户 %s[%s] ajax 验证失效 错误信息为 %s", user.full_name, user.id, repr(exc))
|
2022-11-12 15:08:25 +00:00
|
|
|
|
await self.system.set_challenge(client.uid, gt, challenge)
|
|
|
|
|
url = f"{config.pass_challenge_user_web}?username={context.bot.username}&command=verify>={gt}&challenge={challenge}&uid={client.uid}"
|
2022-11-12 12:59:42 +00:00
|
|
|
|
button = InlineKeyboardMarkup([[InlineKeyboardButton("验证", url=url)]])
|
|
|
|
|
await message.reply_text("请尽快点击下方手动验证", reply_markup=button)
|