mirror of
https://github.com/PaiGramTeam/SIMNet.git
synced 2024-11-22 06:17:57 +00:00
✨ Add transaction client
This commit is contained in:
parent
40d55ea0ee
commit
a92a7f84f8
104
simnet/client/components/transaction.py
Normal file
104
simnet/client/components/transaction.py
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
from typing import Dict, Any, Optional, List, Union
|
||||||
|
from urllib import parse
|
||||||
|
|
||||||
|
from simnet.client.base import BaseClient
|
||||||
|
from simnet.client.routes import YSULOG_URL
|
||||||
|
from simnet.models.genshin.transaction import BaseTransaction, TransactionKind, ItemTransaction, Transaction
|
||||||
|
from simnet.utils.lang import create_short_lang_code
|
||||||
|
|
||||||
|
|
||||||
|
class TransactionClient(BaseClient):
|
||||||
|
"""Transaction component."""
|
||||||
|
|
||||||
|
async def request_transaction(
|
||||||
|
self,
|
||||||
|
endpoint: str,
|
||||||
|
authkey: str,
|
||||||
|
*,
|
||||||
|
method: str = "GET",
|
||||||
|
lang: Optional[str] = None,
|
||||||
|
params: Optional[Dict[str, Any]] = None,
|
||||||
|
) -> Dict[str, Any]:
|
||||||
|
"""Make a request towards the transaction log endpoint.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
endpoint (str): The endpoint to make the request to.
|
||||||
|
authkey (str): The authkey to use for the request.
|
||||||
|
method (str, optional): The HTTP method to use. Defaults to "GET".
|
||||||
|
lang (str, optional): The language to use for the request. Defaults to None.
|
||||||
|
params (Dict[str, Any], optional): The parameters to use for the request. Defaults to None.
|
||||||
|
"""
|
||||||
|
params = dict(params or {})
|
||||||
|
|
||||||
|
base_url = YSULOG_URL.get_url(self.region)
|
||||||
|
url = base_url / endpoint
|
||||||
|
|
||||||
|
params["authkey_ver"] = 1
|
||||||
|
params["sign_type"] = 2
|
||||||
|
params["authkey"] = parse.unquote(authkey)
|
||||||
|
params["lang"] = create_short_lang_code(lang or self.lang)
|
||||||
|
|
||||||
|
return await self.request_lab(url, method=method, params=params)
|
||||||
|
|
||||||
|
async def _get_transaction_page(
|
||||||
|
self,
|
||||||
|
end_id: int,
|
||||||
|
kind: str,
|
||||||
|
authkey: str,
|
||||||
|
*,
|
||||||
|
lang: Optional[str] = None,
|
||||||
|
) -> List[BaseTransaction]:
|
||||||
|
"""Get a single page of transactions.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
end_id (int): The ID of the last transaction to get.
|
||||||
|
kind (str): The kind of transaction to get.
|
||||||
|
authkey (str): The authkey to use for the request.
|
||||||
|
lang (str, optional): The language to use for the request. Defaults to None.
|
||||||
|
"""
|
||||||
|
kind = TransactionKind(kind)
|
||||||
|
endpoint = "get" + kind.value.capitalize() + "Log"
|
||||||
|
|
||||||
|
data = await self.request_transaction(
|
||||||
|
endpoint,
|
||||||
|
lang=lang,
|
||||||
|
authkey=authkey,
|
||||||
|
params=dict(end_id=end_id, size=20),
|
||||||
|
)
|
||||||
|
|
||||||
|
transactions: List[BaseTransaction] = []
|
||||||
|
for trans in data["list"]:
|
||||||
|
model = ItemTransaction if "name" in trans else Transaction
|
||||||
|
transactions.append(model(**trans, kind=kind, lang=lang or self.lang))
|
||||||
|
|
||||||
|
return transactions
|
||||||
|
|
||||||
|
async def transaction_log(
|
||||||
|
self,
|
||||||
|
authkey: str,
|
||||||
|
kind: Optional[Union[str, List[str]]] = None,
|
||||||
|
*,
|
||||||
|
limit: Optional[int] = None,
|
||||||
|
lang: Optional[str] = None,
|
||||||
|
end_id: int = 0,
|
||||||
|
) -> List[BaseTransaction]:
|
||||||
|
"""Get the transaction log of a user.
|
||||||
|
|
||||||
|
Arg:
|
||||||
|
authkey (str): The authkey to use for the request.
|
||||||
|
kind (Union[str, List[str]], optional): The kind of transaction to get. Defaults to None.
|
||||||
|
limit (int, optional): The maximum number of transactions to get. Defaults to None.
|
||||||
|
lang (str, optional): The language to use for the request. Defaults to None.
|
||||||
|
end_id (int, optional): The ID of the last transaction to get. Defaults to 0.
|
||||||
|
"""
|
||||||
|
kinds = kind or ["primogem", "crystal", "resin", "artifact", "weapon"]
|
||||||
|
|
||||||
|
if isinstance(kinds, str):
|
||||||
|
kinds = [kinds]
|
||||||
|
|
||||||
|
iterators: List[BaseTransaction] = []
|
||||||
|
for value in kinds:
|
||||||
|
iterator = await self._get_transaction_page(end_id, value, lang=lang, authkey=authkey)
|
||||||
|
iterators.extend(iterator)
|
||||||
|
|
||||||
|
return iterators[: min(len(iterators), limit)] if limit else iterators
|
@ -5,6 +5,7 @@ from simnet.client.components.chronicle.genshin import GenshinBattleChronicleCli
|
|||||||
from simnet.client.components.daily import DailyRewardClient
|
from simnet.client.components.daily import DailyRewardClient
|
||||||
from simnet.client.components.diary.genshin import GenshinDiaryClient
|
from simnet.client.components.diary.genshin import GenshinDiaryClient
|
||||||
from simnet.client.components.lab import LabClient
|
from simnet.client.components.lab import LabClient
|
||||||
|
from simnet.client.components.transaction import TransactionClient
|
||||||
from simnet.client.components.wish.genshin import GenshinWishClient
|
from simnet.client.components.wish.genshin import GenshinWishClient
|
||||||
from simnet.utils.enum_ import Game
|
from simnet.utils.enum_ import Game
|
||||||
|
|
||||||
@ -18,7 +19,8 @@ class GenshinClient(
|
|||||||
AuthClient,
|
AuthClient,
|
||||||
DailyRewardClient,
|
DailyRewardClient,
|
||||||
LabClient,
|
LabClient,
|
||||||
|
TransactionClient,
|
||||||
):
|
):
|
||||||
"""A simple http client for StarRail endpoints."""
|
"""A simple http client for Genshin endpoints."""
|
||||||
|
|
||||||
game: Optional[Game] = Game.GENSHIN
|
game: Optional[Game] = Game.GENSHIN
|
||||||
|
@ -5,13 +5,20 @@ from simnet.client.components.chronicle.genshin import GenshinBattleChronicleCli
|
|||||||
from simnet.client.components.daily import DailyRewardClient
|
from simnet.client.components.daily import DailyRewardClient
|
||||||
from simnet.client.components.diary.genshin import GenshinDiaryClient
|
from simnet.client.components.diary.genshin import GenshinDiaryClient
|
||||||
from simnet.client.components.lab import LabClient
|
from simnet.client.components.lab import LabClient
|
||||||
|
from simnet.client.components.transaction import TransactionClient
|
||||||
from simnet.client.components.wish.genshin import GenshinWishClient
|
from simnet.client.components.wish.genshin import GenshinWishClient
|
||||||
from simnet.utils.enum_ import Region
|
from simnet.utils.enum_ import Region
|
||||||
from simnet.utils.types import CookieTypes, HeaderTypes, TimeoutTypes
|
from simnet.utils.types import CookieTypes, HeaderTypes, TimeoutTypes
|
||||||
|
|
||||||
|
|
||||||
class GenshinClient(
|
class GenshinClient(
|
||||||
GenshinBattleChronicleClient, GenshinWishClient, GenshinDiaryClient, AuthClient, DailyRewardClient, LabClient
|
GenshinBattleChronicleClient,
|
||||||
|
GenshinWishClient,
|
||||||
|
GenshinDiaryClient,
|
||||||
|
AuthClient,
|
||||||
|
DailyRewardClient,
|
||||||
|
LabClient,
|
||||||
|
TransactionClient,
|
||||||
):
|
):
|
||||||
def __init__(
|
def __init__(
|
||||||
self,
|
self,
|
||||||
|
@ -27,6 +27,7 @@ __all__ = (
|
|||||||
"INFO_LEDGER_URL",
|
"INFO_LEDGER_URL",
|
||||||
"HK4E_URL",
|
"HK4E_URL",
|
||||||
"CODE_URL",
|
"CODE_URL",
|
||||||
|
"YSULOG_URL",
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -255,6 +256,10 @@ INFO_LEDGER_URL = GameRoute(
|
|||||||
),
|
),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
YSULOG_URL = InternationalRoute(
|
||||||
|
overseas="https://hk4e-api-os.hoyoverse.com/ysulog/api/",
|
||||||
|
chinese="https://hk4e-api.mihoyo.com/ysulog/api/",
|
||||||
|
)
|
||||||
|
|
||||||
HK4E_URL = Route("https://sg-hk4e-api.hoyoverse.com/common/hk4e_global/")
|
HK4E_URL = Route("https://sg-hk4e-api.hoyoverse.com/common/hk4e_global/")
|
||||||
|
|
||||||
|
54
simnet/models/genshin/transaction.py
Normal file
54
simnet/models/genshin/transaction.py
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from enum import Enum
|
||||||
|
from typing import Literal
|
||||||
|
|
||||||
|
from pydantic import Field
|
||||||
|
|
||||||
|
from simnet.models.base import APIModel
|
||||||
|
|
||||||
|
__all__ = ("BaseTransaction", "ItemTransaction", "Transaction", "TransactionKind")
|
||||||
|
|
||||||
|
|
||||||
|
class TransactionKind(str, Enum):
|
||||||
|
"""Possible kind of transaction.
|
||||||
|
|
||||||
|
Attributes:
|
||||||
|
PRIMOGEM: Primogem currency.
|
||||||
|
CRYSTAL: Genesis crystal currency.
|
||||||
|
RESIN: Resin currency.
|
||||||
|
ARTIFACT: Artifact items from domains.
|
||||||
|
WEAPON: Weapon items from domains and wishes.
|
||||||
|
"""
|
||||||
|
|
||||||
|
PRIMOGEM = "primogem"
|
||||||
|
CRYSTAL = "crystal"
|
||||||
|
RESIN = "resin"
|
||||||
|
ARTIFACT = "artifact"
|
||||||
|
WEAPON = "weapon"
|
||||||
|
|
||||||
|
|
||||||
|
class BaseTransaction(APIModel):
|
||||||
|
"""Genshin transaction."""
|
||||||
|
|
||||||
|
kind: TransactionKind
|
||||||
|
|
||||||
|
id: int
|
||||||
|
uid: int
|
||||||
|
time: datetime
|
||||||
|
amount: int = Field(alias="add_num")
|
||||||
|
reason_id: int = Field(alias="reason")
|
||||||
|
|
||||||
|
|
||||||
|
class Transaction(BaseTransaction):
|
||||||
|
"""Genshin transaction of currency."""
|
||||||
|
|
||||||
|
kind: Literal[TransactionKind.PRIMOGEM, TransactionKind.CRYSTAL, TransactionKind.RESIN]
|
||||||
|
|
||||||
|
|
||||||
|
class ItemTransaction(BaseTransaction):
|
||||||
|
"""Genshin transaction of artifacts or weapons."""
|
||||||
|
|
||||||
|
kind: Literal[TransactionKind.ARTIFACT, TransactionKind.WEAPON]
|
||||||
|
|
||||||
|
name: str
|
||||||
|
rarity: int = Field(alias="rank")
|
Loading…
Reference in New Issue
Block a user