Refactor auth client and add test

This commit is contained in:
洛水居室 2023-05-09 16:06:32 +08:00 committed by GitHub
parent b74b9aef33
commit 912a41fe45
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 154 additions and 37 deletions

View File

@ -20,26 +20,37 @@ class AuthClient(BaseClient):
different authentication tokens and keys.
"""
async def get_stoken_by_login_ticket(self) -> bool:
async def get_stoken_by_login_ticket(
self, login_ticket: Optional[str] = None, account_id: Optional[int] = None
) -> Optional[str]:
"""
Retrieves a super ticket (`stoken`) using a login ticket (`login_ticket`) .
Args:
login_ticket (Optional[str]): The login ticket to use to retrieve the super ticket. If not provided, the
`login_ticket` cookie value will be used.
account_id (Optional[int]): The account ID to use to retrieve the super ticket. If not provided, the
`account_id` attribute value will be used.
Returns:
bool: `True` if the super ticket successfully retrieved, otherwise `False`.
Optional[str]: The retrieved super ticket (`stoken`).
Raises:
RuntimeError: This method is only available for the Chinese region.
ValueError: If `login_ticket` is not found in the cookies.
ValueError: If the `login_ticket` argument is `None`, or if the `account_id` argument is `None`.
"""
if self.region != Region.CHINESE:
raise RuntimeError("This method is only available for the Chinese region.")
url = AUTH_URL.get_url(Region.CHINESE).join("getMultiTokenByLoginTicket")
login_ticket = self.cookies.get("login_ticket")
url = AUTH_URL.get_url(Region.CHINESE) / "getMultiTokenByLoginTicket"
login_ticket = login_ticket or self.cookies.get("login_ticket")
account_id = account_id or self.account_id
if login_ticket is None:
raise ValueError("login_ticket not found in cookies.")
raise ValueError("The 'login_ticket' argument cannot be None.")
if account_id is None:
raise ValueError("The 'account_id' argument cannot be None.")
params = {
"login_ticket": login_ticket,
"uid": self.account_id,
"uid": account_id,
"token_types": 3,
}
data = await self.request_lab(url, params=params)
@ -51,73 +62,89 @@ class AuthClient(BaseClient):
self.cookies[name] = token
stoken = self.cookies.get("stoken")
stuid = self.cookies.get("stuid")
if stoken:
if stuid:
if stoken and stuid:
self.cookies["stuid"] = self.account_id
return True
return False
return stoken
async def get_cookie_token_by_stoken(self) -> bool:
async def get_cookie_token_by_stoken(
self, stoken: Optional[str] = None, account_id: Optional[int] = None
) -> Optional[str]:
"""
Retrieves a cookie token (`cookie_token`) using a super ticket (`stoken`).
Args:
stoken (Optional[str]): The super ticket to use to retrieve the cookie token. If not provided, the
`stoken` cookie value will be used.
account_id (Optional[int]): The account ID to use to retrieve the cookie token. If not provided, the
`account_id` attribute value will be used.
Returns:
bool: `True` if the cookie token was successfully retrieved, otherwise `False`.
Optional[str]: The retrieved cookie token (`cookie_token`).
Raises:
RuntimeError: This method is only available for the Chinese region.
ValueError: If `stoken` is not found in the cookies.
ValueError: If the `login_ticket` argument is `None`, or if the `account_id` argument is `None`.
"""
if self.region != Region.CHINESE:
raise RuntimeError("This method is only available for the Chinese region.")
stoken = self.cookies.get("stoken")
stoken = stoken or self.cookies.get("stoken")
account_id = account_id or self.account_id
if stoken is None:
raise ValueError("stoken not found in cookies.")
raise ValueError("The 'stoken' argument cannot be None.")
if account_id is None:
raise ValueError("The 'account_id' argument cannot be None.")
url = GET_COOKIES_TOKEN_BY_STOKEN_URL.get_url(Region.CHINESE)
params = {
"stoken": stoken,
"uid": self.account_id,
"uid": account_id,
}
data = await self.request_lab(url, params=params)
cookie_token = data.get("cookie_token", "")
cookie_token = data.get("cookie_token")
if cookie_token:
self.cookies["cookie_token"] = cookie_token
self.cookies["account_id"] = self.account_id
return True
return False
return cookie_token
async def get_ltoken_by_stoken(self) -> bool:
async def get_ltoken_by_stoken(
self, stoken: Optional[str] = None, account_id: Optional[int] = None
) -> Optional[str]:
"""
Retrieves a login token (`ltoken`) using a super ticket (`stoken`).
Args:
stoken (Optional[str]): The super ticket to use to retrieve the cookie token. If not provided, the
`stoken` cookie value will be used.
account_id (Optional[int]): The account ID to use to retrieve the cookie token. If not provided, the
`account_id` attribute value will be used.
Returns:
bool: `True` if the login token was successfully retrieved, otherwise `False`.
Optional[str]: The retrieved cookie token (`cookie_token`).
Raises:
RuntimeError: This method is only available for the Chinese region.
ValueError: If `stoken` is not found in the cookies.
ValueError: If the `login_ticket` argument is `None`, or if the `account_id` argument is `None`.
"""
if self.region != Region.CHINESE:
raise RuntimeError("This method is only available for the Chinese region.")
stoken = self.cookies.get("stoken")
stoken = stoken or self.cookies.get("stoken")
account_id = account_id or self.account_id
if stoken is None:
raise ValueError("stoken not found in cookies.")
raise ValueError("The 'stoken' argument cannot be None.")
if account_id is None:
raise ValueError("The 'account_id' argument cannot be None.")
url = GET_LTOKEN_BY_STOKEN_URL.get_url(Region.CHINESE)
params = {
"stoken": stoken,
"uid": self.account_id,
"uid": account_id,
}
data = await self.request_lab(url, params=params)
ltoken = data.get("ltoken", "")
if ltoken:
self.cookies["ltoken"] = ltoken
self.cookies["ltuid"] = self.account_id
return True
return False
return ltoken
async def get_authkey_by_stoken(
self, game_biz: str, region: str, auth_appid: str
) -> Optional[str]:
async def get_authkey_by_stoken(self, game_biz: str, region: str, auth_appid: str) -> Optional[str]:
"""
Get the auth key (`authkey`) for a game and region using a super ticket (`stoken`).
@ -128,17 +155,22 @@ class AuthClient(BaseClient):
For example, to request wish records, use `webview_gacha`.
Returns:
str or None: The authentication key, or None if not found.
Optional[str]: The authentication key, or None if not found.
Raises:
RuntimeError: This method is only available for the Chinese region.
ValueError: If `stoken` is not found in the cookies.
ValueError: If `stoken` is not found in the cookies or `player_id` not found.
"""
if self.region != Region.CHINESE:
raise RuntimeError("This method is only available for the Chinese region.")
stoken = self.cookies.get("stoken")
if stoken is None:
raise ValueError("stoken not found in cookies.")
stuid = self.cookies.get("stuid")
if stuid is None and self.account_id is None:
raise ValueError("account_id or stuid not found")
if self.account_id is not None and stuid is None:
self.cookies.set("stuid", str(self.account_id))
url = AUTH_KEY_URL.get_url(self.region)
json = {
"auth_appid": auth_appid,
@ -150,7 +182,7 @@ class AuthClient(BaseClient):
return data.get("authkey")
async def get_hk4e_token_by_cookie_token(
self, game_biz: str, region: str
self, game_biz: str, region: str, player_id: Optional[int] = None
) -> NoReturn:
"""
Get HK4E token (`hk4e_token`) using cookie token (`cookie_token`).
@ -159,6 +191,8 @@ class AuthClient(BaseClient):
Args:
game_biz (str): The name of the game.
region (str): The region in which the game is registered.
player_id (Optional[int]): The player ID to use to retrieve the HK4E token. If not provided, the `player_id`
attribute value will be used.
Raises:
ValueError: If `cookie_token` is not found in the cookies.
@ -169,7 +203,7 @@ class AuthClient(BaseClient):
url = HK4E_LOGIN_URL.get_url(self.region)
json = {
"game_biz": game_biz,
"uid": self.player_id,
"uid": self.player_id or player_id,
"region": region,
}
await self.request_lab(url, data=json)

View File

@ -2,3 +2,5 @@ COOKIES=
ACCOUNT_ID=
GENSHIN_PLAYER_ID=
STARRAIL_PLAYER_ID=
# STOKEN=
# LOGIN_TICKET=

View File

@ -2,6 +2,7 @@ import asyncio
import os
import warnings
from pathlib import Path
from typing import Optional
import pytest
from dotenv import load_dotenv
@ -58,3 +59,15 @@ def account_id() -> int: # skipcq: PY-D0003
if not _account_id:
pytest.exit("No account id set", 1)
return int(_account_id)
@pytest.fixture(scope="session")
def stoken() -> Optional[str]: # skipcq: PY-D0003
_stoken = os.environ.get("STOKEN")
return _stoken
@pytest.fixture(scope="session")
def login_ticket() -> Optional[str]: # skipcq: PY-D0003
_login_ticket = os.environ.get("LOGIN_TICKET")
return _login_ticket

68
tests/test_auth_client.py Normal file
View File

@ -0,0 +1,68 @@
from typing import TYPE_CHECKING
import pytest
import pytest_asyncio
from simnet.client.components.auth import AuthClient
from simnet.utils.enum_ import Region
from simnet.utils.player import recognize_genshin_server
if TYPE_CHECKING:
from simnet.client.cookies import Cookies
@pytest_asyncio.fixture
async def auth_client(account_id: int, cookies: "Cookies"):
async with AuthClient(
cookies=cookies,
account_id=account_id,
region=Region.CHINESE,
) as client_instance:
yield client_instance
@pytest.mark.asyncio
class TestAuthClient:
@staticmethod
async def test_get_hk4e_token_by_cookie_token(auth_client: "AuthClient", genshin_player_id: int):
await auth_client.get_hk4e_token_by_cookie_token(
"hk4e_cn", recognize_genshin_server(genshin_player_id), player_id=genshin_player_id
)
hk4e_token = auth_client.client.cookies.get("e_hk4e_token")
assert hk4e_token is not None
@staticmethod
async def test_get_stoken_by_login_ticket(auth_client: "AuthClient", login_ticket: str, account_id: int):
if login_ticket is None:
pytest.skip("Test case test_get_stoken_by_login_ticket skipped: Parameter login_ticket is None")
stoken = await auth_client.get_stoken_by_login_ticket(login_ticket, account_id)
assert stoken is not None
@staticmethod
async def test_get_cookie_token_by_stoken(auth_client: "AuthClient", stoken: str, account_id: int):
if stoken is None:
pytest.skip("Test case test_get_cookie_token_by_stoken skipped: Parameter stoken is None")
cookie_token = await auth_client.get_cookie_token_by_stoken(stoken, account_id)
assert cookie_token is not None
@staticmethod
async def test_get_ltoken_by_stoken(auth_client: "AuthClient", stoken: str, account_id: int):
if stoken is None:
pytest.skip("Test case test_get_ltoken_by_stoken skipped: Parameter stoken is None")
ltoken = await auth_client.get_ltoken_by_stoken(stoken, account_id)
assert ltoken is not None
@staticmethod
async def test_get_authkey_by_stoken(stoken: str, account_id: int, genshin_player_id: int):
if stoken is None:
pytest.skip("Test case test_get_authkey_by_stoken skipped: Parameter stoken is None")
async with AuthClient(
cookies={"stoken": stoken},
player_id=genshin_player_id,
account_id=account_id,
region=Region.CHINESE,
) as client_instance:
authkey = await client_instance.get_authkey_by_stoken(
"hk4e_cn", recognize_genshin_server(genshin_player_id), "webview_gacha"
)
assert authkey is not None