MibooGram/core/dependence/assets.py

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

387 lines
14 KiB
Python
Raw Normal View History

import asyncio
from pathlib import Path
from ssl import SSLZeroReturnError
from typing import Optional, List, Dict
from aiofiles import open as async_open
from httpx import AsyncClient, HTTPError
from core.base_service import BaseService
from modules.wiki.base import WikiModel
from modules.wiki.models.avatar import Avatar
from modules.wiki.models.weapon import Weapon
from modules.wiki.models.buddy import Buddy
from modules.wiki.models.equipment_suit import EquipmentSuit
from utils.const import PROJECT_ROOT
from utils.log import logger
from utils.typedefs import StrOrURL, StrOrInt
ASSETS_PATH = PROJECT_ROOT.joinpath("resources/assets")
ASSETS_PATH.mkdir(exist_ok=True, parents=True)
DATA_MAP = {
"avatar": WikiModel.BASE_URL + "avatars.json",
"weapon": WikiModel.BASE_URL + "weapons.json",
"buddy": WikiModel.BASE_URL + "buddy.json",
"equipment_suit": WikiModel.BASE_URL + "equipment_suits.json",
}
2024-12-18 09:10:21 +00:00
def choose_path_by_url(url: str, png_path: Path, webp_path: Path) -> Path:
ext = url.split(".")[-1].lower()
if ext == "png":
return png_path
if ext == "webp":
return webp_path
return png_path
class AssetsServiceError(Exception):
pass
class AssetsCouldNotFound(AssetsServiceError):
def __init__(self, message: str, target: str):
self.message = message
self.target = target
super().__init__(f"{message}: target={target}")
class _AssetsService:
client: Optional[AsyncClient] = None
def __init__(self, client: Optional[AsyncClient] = None) -> None:
self.client = client
async def _download(self, url: StrOrURL, path: Path, retry: int = 5) -> Optional[Path]:
"""从 url 下载图标至 path"""
if not url:
return None
logger.debug("正在从 %s 下载图标至 %s", url, path)
headers = None
for time in range(retry):
try:
response = await self.client.get(url, follow_redirects=False, headers=headers)
except Exception as error: # pylint: disable=W0703
if not isinstance(error, (HTTPError, SSLZeroReturnError)):
logger.error(error) # 打印未知错误
if time != retry - 1: # 未达到重试次数
await asyncio.sleep(1)
else:
raise error
continue
if response.status_code != 200: # 判定页面是否正常
return None
async with async_open(path, "wb") as file:
await file.write(response.content) # 保存图标
return path.resolve()
class _AvatarAssets(_AssetsService):
path: Path
data: List[Avatar]
name_map: Dict[str, Avatar]
id_map: Dict[int, Avatar]
def __init__(self, client: Optional[AsyncClient] = None) -> None:
super().__init__(client)
self.path = ASSETS_PATH.joinpath("agent")
self.path.mkdir(exist_ok=True, parents=True)
async def initialize(self):
logger.info("正在初始化角色素材图标")
html = await self.client.get(DATA_MAP["avatar"])
self.data = [Avatar(**data) for data in html.json()]
self.name_map = {icon.name: icon for icon in self.data}
self.id_map = {icon.id: icon for icon in self.data}
tasks = []
for icon in self.data:
base_path = self.path / f"{icon.id}"
base_path.mkdir(exist_ok=True, parents=True)
2024-12-18 09:10:21 +00:00
def _get_path(name: str) -> tuple[Path, Path]:
path = base_path / f"{name}.png"
return path, path.with_suffix(".webp")
for i in (("gacha", icon.gacha), ("icon", icon.icon_), ("square", icon.square), ("normal", icon.normal)):
png_path, webp_path = _get_path(i[0])
if not png_path.exists() and not webp_path.exists() and i[1]:
tasks.append(self._download(i[1], choose_path_by_url(i[1], png_path, webp_path)))
if len(tasks) >= 100:
await asyncio.gather(*tasks)
tasks = []
if tasks:
await asyncio.gather(*tasks)
logger.info("角色素材图标初始化完成")
2024-12-18 09:10:21 +00:00
def get_path(self, icon: Avatar, name: str) -> Path:
path = self.path / f"{icon.id}"
path.mkdir(exist_ok=True, parents=True)
2024-12-18 09:10:21 +00:00
p1 = path / f"{name}.png"
if p1.exists():
return p1
p2 = path / f"{name}.webp"
if p2.exists():
return p2
return p1
def get_by_id(self, id_: int) -> Optional[Avatar]:
return self.id_map.get(id_, None)
def get_by_name(self, name: str) -> Optional[Avatar]:
return self.name_map.get(name, None)
def get_target(self, target: StrOrInt, second_target: StrOrInt = None) -> Avatar:
data = None
if isinstance(target, int):
data = self.get_by_id(target)
elif isinstance(target, str):
data = self.get_by_name(target)
if data is None:
if second_target:
return self.get_target(second_target)
raise AssetsCouldNotFound("角色素材图标不存在", target)
return data
def gacha(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
icon = self.get_target(target, second_target)
return self.get_path(icon, "gacha")
def icon(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
icon = self.get_target(target, second_target)
return self.get_path(icon, "icon")
2024-08-16 03:39:47 +00:00
def square(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
icon = self.get_target(target, second_target)
return self.get_path(icon, "square")
def normal(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
icon = self.get_target(target, second_target)
return self.get_path(icon, "normal")
class _WeaponAssets(_AssetsService):
path: Path
data: List[Weapon]
name_map: Dict[str, Weapon]
id_map: Dict[int, Weapon]
def __init__(self, client: Optional[AsyncClient] = None) -> None:
super().__init__(client)
self.path = ASSETS_PATH.joinpath("engines")
self.path.mkdir(exist_ok=True, parents=True)
async def initialize(self):
logger.info("正在初始化武器素材图标")
html = await self.client.get(DATA_MAP["weapon"])
self.data = [Weapon(**data) for data in html.json()]
self.name_map = {icon.name: icon for icon in self.data}
self.id_map = {icon.id: icon for icon in self.data}
tasks = []
for icon in self.data:
base_path = self.path / f"{icon.id}"
base_path.mkdir(exist_ok=True, parents=True)
icon_path = base_path / "icon.webp"
if not icon_path.exists():
tasks.append(self._download(icon.icon, icon_path))
if len(tasks) >= 100:
await asyncio.gather(*tasks)
tasks = []
if tasks:
await asyncio.gather(*tasks)
logger.info("武器素材图标初始化完成")
def get_path(self, icon: Weapon, name: str) -> Path:
path = self.path / f"{icon.id}"
path.mkdir(exist_ok=True, parents=True)
return path / f"{name}.webp"
def get_by_id(self, id_: int) -> Optional[Weapon]:
return self.id_map.get(id_, None)
def get_by_name(self, name: str) -> Optional[Weapon]:
return self.name_map.get(name, None)
def get_target(self, target: StrOrInt, second_target: StrOrInt = None) -> Optional[Weapon]:
if isinstance(target, int):
return self.get_by_id(target)
elif isinstance(target, str):
return self.get_by_name(target)
if second_target:
return self.get_target(second_target)
raise AssetsCouldNotFound("武器素材图标不存在", target)
def icon(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
icon = self.get_target(target, second_target)
return self.get_path(icon, "icon")
class _BuddyAssets(_AssetsService):
path: Path
data: List[Buddy]
id_map: Dict[int, Buddy]
name_map: Dict[str, Buddy]
def __init__(self, client: Optional[AsyncClient] = None) -> None:
super().__init__(client)
self.path = ASSETS_PATH.joinpath("buddy")
self.path.mkdir(exist_ok=True, parents=True)
async def initialize(self):
logger.info("正在初始化邦布素材图标")
html = await self.client.get(DATA_MAP["buddy"])
self.data = [Buddy(**data) for data in html.json()]
self.id_map = {icon.id: icon for icon in self.data}
self.name_map = {icon.name: icon for icon in self.data}
tasks = []
for icon in self.data:
webp_path = self.path / f"{icon.id}.webp"
png_path = self.path / f"{icon.id}.png"
2024-08-16 03:39:47 +00:00
square_path = self.path / f"{icon.id}_square.png"
if not webp_path.exists() and icon.webp:
tasks.append(self._download(icon.webp, webp_path))
if not png_path.exists() and icon.png:
tasks.append(self._download(icon.png, png_path))
2024-08-16 03:39:47 +00:00
if not square_path.exists() and icon.square:
tasks.append(self._download(icon.square, square_path))
if len(tasks) >= 100:
await asyncio.gather(*tasks)
tasks = []
if tasks:
await asyncio.gather(*tasks)
logger.info("邦布素材图标初始化完成")
2024-08-16 03:39:47 +00:00
def get_path(self, icon: Buddy, ext: str, square: bool = False) -> Path:
square_str = "_square" if square else ""
path = self.path / f"{icon.id}{square_str}.{ext}"
return path
def get_by_id(self, id_: int) -> Optional[Buddy]:
return self.id_map.get(id_, None)
def get_by_name(self, name: str) -> Optional[Buddy]:
return self.name_map.get(name, None)
def get_target(self, target: StrOrInt, second_target: StrOrInt = None) -> Optional[Buddy]:
if isinstance(target, int):
return self.get_by_id(target)
elif isinstance(target, str):
return self.get_by_name(target)
if second_target:
return self.get_target(second_target)
raise AssetsCouldNotFound("邦布素材图标不存在", target)
def webp(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
icon = self.get_target(target, second_target)
return self.get_path(icon, "webp")
def png(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
icon = self.get_target(target, second_target)
return self.get_path(icon, "png")
def icon(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
icon = self.get_target(target, second_target)
webp_path = self.get_path(icon, "webp")
png_path = self.get_path(icon, "png")
if webp_path.exists():
return webp_path
if png_path.exists():
return png_path
raise AssetsCouldNotFound("邦布素材图标不存在", target)
2024-08-16 03:39:47 +00:00
def square(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
icon = self.get_target(target, second_target)
return self.get_path(icon, "png", square=True)
class _EquipmentSuitAssets(_AssetsService):
path: Path
data: List[EquipmentSuit]
id_map: Dict[int, EquipmentSuit]
name_map: Dict[str, EquipmentSuit]
def __init__(self, client: Optional[AsyncClient] = None) -> None:
super().__init__(client)
self.path = ASSETS_PATH.joinpath("equipment_suit")
self.path.mkdir(exist_ok=True, parents=True)
async def initialize(self):
logger.info("正在初始化驱动盘素材图标")
html = await self.client.get(DATA_MAP["equipment_suit"])
self.data = [EquipmentSuit(**data) for data in html.json()]
self.id_map = {theme.id: theme for theme in self.data}
self.name_map = {theme.name: theme for theme in self.data}
tasks = []
for theme in self.data:
path = self.path / f"{theme.id}.webp"
if not path.exists():
tasks.append(self._download(theme.icon, path))
if len(tasks) >= 100:
await asyncio.gather(*tasks)
tasks = []
if tasks:
await asyncio.gather(*tasks)
logger.info("驱动盘素材图标初始化完成")
def get_path(self, theme: EquipmentSuit, ext: str) -> Path:
path = self.path / f"{theme.id}.{ext}"
return path
def get_by_id(self, id_: int) -> Optional[EquipmentSuit]:
return self.id_map.get(id_, None)
def get_by_name(self, name_: str) -> Optional[EquipmentSuit]:
return self.name_map.get(name_, None)
def get_target(self, target: StrOrInt, second_target: StrOrInt = None) -> Optional[EquipmentSuit]:
if isinstance(target, int):
return self.get_by_id(target)
elif isinstance(target, str):
return self.get_by_name(target)
if second_target:
return self.get_target(second_target)
raise AssetsCouldNotFound("驱动盘素材图标不存在", target)
def icon(self, target: StrOrInt, second_target: StrOrInt = None) -> Path:
theme = self.get_target(target, second_target)
webp_path = self.get_path(theme, "webp")
if webp_path.exists():
return webp_path
raise AssetsCouldNotFound("驱动盘素材图标不存在", target)
class AssetsService(BaseService.Dependence):
"""asset服务
用于储存和管理 asset :
当对应的 asset (如某角色图标)不存在时该服务会先查找本地
若本地不存在则从网络上下载若存在则返回其路径
"""
client: Optional[AsyncClient] = None
avatar: _AvatarAssets
"""角色"""
weapon: _WeaponAssets
"""武器"""
buddy: _BuddyAssets
"""邦布"""
equipment_suit: _EquipmentSuitAssets
"""驱动盘"""
def __init__(self):
self.client = AsyncClient(timeout=60.0)
self.avatar = _AvatarAssets(self.client)
self.weapon = _WeaponAssets(self.client)
self.buddy = _BuddyAssets(self.client)
self.equipment_suit = _EquipmentSuitAssets(self.client)
async def initialize(self): # pylint: disable=W0221
await self.avatar.initialize()
await self.weapon.initialize()
await self.buddy.initialize()
await self.equipment_suit.initialize()