zhenxun_bot/plugins/genshin/query_resource_points/query_resource.py

273 lines
10 KiB
Python
Raw Normal View History

2022-05-22 18:08:06 +08:00
from pathlib import Path
2021-07-30 21:21:51 +08:00
from typing import Tuple, Optional, List
2022-09-21 21:57:25 +08:00
from configs.path_config import IMAGE_PATH, TEXT_PATH, TEMP_PATH
2021-07-30 21:21:51 +08:00
from utils.message_builder import image
2021-06-21 09:53:33 +08:00
from services.log import logger
2021-12-16 11:16:28 +08:00
from utils.image_utils import BuildImage
2021-07-30 21:21:51 +08:00
from asyncio.exceptions import TimeoutError
from asyncio import Semaphore
from utils.image_utils import is_valid
2021-11-23 21:44:59 +08:00
from utils.http_utils import AsyncHttpx
2021-12-01 14:03:34 +08:00
from httpx import ConnectTimeout
2021-11-23 21:44:59 +08:00
from .map import Map
import asyncio
2021-06-15 10:57:08 +08:00
import nonebot
2021-07-30 21:21:51 +08:00
import os
2021-08-04 15:19:45 +08:00
2021-07-30 21:21:51 +08:00
try:
import ujson as json
except ModuleNotFoundError:
import json
2021-06-15 10:57:08 +08:00
driver: nonebot.Driver = nonebot.get_driver()
2021-05-20 19:34:27 +08:00
2021-07-30 21:21:51 +08:00
LABEL_URL = "https://api-static.mihoyo.com/common/blackboard/ys_obc/v1/map/label/tree?app_sn=ys_obc"
POINT_LIST_URL = "https://api-static.mihoyo.com/common/blackboard/ys_obc/v1/map/point/list?map_id=2&app_sn=ys_obc"
MAP_URL = "https://api-static.mihoyo.com/common/map_user/ys_obc/v1/map/info?map_id=2&app_sn=ys_obc&lang=zh-cn"
2022-02-19 18:20:19 +08:00
icon_path = IMAGE_PATH / "genshin" / "genshin_icon"
map_path = IMAGE_PATH / "genshin" / "map"
resource_label_file = TEXT_PATH / "genshin" / "resource_label_file.json"
resource_point_file = TEXT_PATH / "genshin" / "resource_point_file.json"
resource_type_file = TEXT_PATH / "genshin" / "resource_type_file.json"
2021-07-30 21:21:51 +08:00
# 地图中心坐标
CENTER_POINT: Optional[Tuple[int, int]] = None
resource_name_list: List[str] = []
2021-08-04 15:19:45 +08:00
MAP_RATIO = 0.5
2021-07-30 21:21:51 +08:00
# 查找资源
async def query_resource(resource_name: str) -> str:
2021-08-04 15:19:45 +08:00
global CENTER_POINT
2021-07-30 21:21:51 +08:00
planning_route: bool = False
2021-08-17 23:17:08 +08:00
if resource_name and resource_name[-2:] in ["路径", "路线"]:
2021-07-30 21:21:51 +08:00
resource_name = resource_name[:-2].strip()
planning_route = True
if not resource_name or resource_name not in resource_name_list:
2021-10-03 14:24:07 +08:00
# return f"未查找到 {resource_name} 资源,可通过 “原神资源列表” 获取全部资源名称.."
2022-02-19 20:12:51 +08:00
return ""
2021-08-04 15:19:45 +08:00
map_ = Map(
resource_name, CENTER_POINT, planning_route=planning_route, ratio=MAP_RATIO
)
2021-07-30 21:21:51 +08:00
count = map_.get_resource_count()
2021-08-04 15:19:45 +08:00
rand = await asyncio.get_event_loop().run_in_executor(
None, map_.generate_resource_icon_in_map
)
return (
2022-09-21 21:57:25 +08:00
f"{image(TEMP_PATH / f'genshin_map_{rand}.png')}"
2021-08-04 15:19:45 +08:00
f"\n\n{resource_name} 一共找到 {count} 个位置点\n※ 数据来源于米游社wiki"
)
2021-07-30 21:21:51 +08:00
# 原神资源列表
def get_resource_type_list():
with open(resource_type_file, "r", encoding="utf8") as f:
data = json.load(f)
temp = {}
for id_ in data.keys():
temp[data[id_]["name"]] = []
for x in data[id_]["children"]:
temp[data[id_]["name"]].append(x["name"])
2021-05-20 19:34:27 +08:00
2021-07-30 21:21:51 +08:00
mes = "当前资源列表如下:\n"
2021-05-20 19:34:27 +08:00
2021-07-30 21:21:51 +08:00
for resource_type in temp.keys():
mes += f"{resource_type}{''.join(temp[resource_type])}\n"
return mes
2021-05-20 19:34:27 +08:00
2021-06-15 10:57:08 +08:00
2021-10-03 14:24:07 +08:00
def check_resource_exists(resource: str) -> bool:
"""
检查资源是否存在
:param resource: 资源名称
"""
2022-02-19 20:12:51 +08:00
resource = resource.replace("路径", "").replace("路线", "")
2021-10-03 14:24:07 +08:00
return resource in resource_name_list
2021-06-15 10:57:08 +08:00
@driver.on_startup
2021-07-30 21:21:51 +08:00
async def init(flag: bool = False):
global CENTER_POINT, resource_name_list
2021-10-03 14:24:07 +08:00
try:
semaphore = asyncio.Semaphore(10)
2021-11-23 21:44:59 +08:00
await download_map_init(semaphore, flag)
await download_resource_data(semaphore)
await download_resource_type()
if not CENTER_POINT:
2022-02-19 20:03:40 +08:00
if resource_label_file.exists():
2022-02-19 20:12:51 +08:00
CENTER_POINT = json.load(
open(resource_label_file, "r", encoding="utf8")
)["CENTER_POINT"]
2022-02-19 20:03:40 +08:00
if resource_label_file.exists():
with open(resource_type_file, "r", encoding="utf8") as f:
data = json.load(f)
2022-02-19 20:12:51 +08:00
for id_ in data:
for x in data[id_]["children"]:
resource_name_list.append(x["name"])
2021-10-03 14:24:07 +08:00
except TimeoutError:
2022-02-19 20:12:51 +08:00
logger.warning("原神资源查询信息初始化超时....")
2021-07-30 21:21:51 +08:00
# 图标及位置资源
2021-11-23 21:44:59 +08:00
async def download_resource_data(semaphore: Semaphore):
2021-07-30 21:21:51 +08:00
icon_path.mkdir(parents=True, exist_ok=True)
resource_label_file.parent.mkdir(parents=True, exist_ok=True)
2021-06-21 09:53:33 +08:00
try:
2022-02-19 20:03:40 +08:00
response = await AsyncHttpx.get(POINT_LIST_URL, timeout=10)
2021-11-23 21:44:59 +08:00
if response.status_code == 200:
data = response.json()
if data["message"] == "OK":
data = data["data"]
for lst in ["label_list", "point_list"]:
resource_data = {"CENTER_POINT": CENTER_POINT}
tasks = []
file = (
resource_label_file
if lst == "label_list"
else resource_point_file
)
for x in data[lst]:
id_ = x["id"]
if lst == "label_list":
img_url = x["icon"]
tasks.append(
asyncio.ensure_future(
download_image(
img_url,
2022-05-22 18:08:06 +08:00
icon_path / f"{id_}.png",
2021-11-23 21:44:59 +08:00
semaphore,
True,
2021-07-30 21:21:51 +08:00
)
)
2021-11-23 21:44:59 +08:00
)
resource_data[id_] = x
await asyncio.gather(*tasks)
with open(file, "w", encoding="utf8") as f:
json.dump(resource_data, f, ensure_ascii=False, indent=4)
2021-07-30 21:21:51 +08:00
else:
2021-11-23 21:44:59 +08:00
logger.warning(f'获取原神资源失败 msg: {data["message"]}')
else:
logger.warning(f"获取原神资源失败 code{response.status_code}")
2021-12-01 14:03:34 +08:00
except (TimeoutError, ConnectTimeout):
2021-07-30 21:21:51 +08:00
logger.warning("获取原神资源数据超时...已再次尝试...")
2021-11-23 21:44:59 +08:00
await download_resource_data(semaphore)
2021-12-01 14:03:34 +08:00
except Exception as e:
logger.error(f"获取原神资源数据未知错误 {type(e)}{e}")
2021-07-30 21:21:51 +08:00
# 下载原神地图并拼图
2022-02-19 20:12:51 +08:00
async def download_map_init(semaphore: Semaphore, flag: bool = False):
2021-08-04 15:19:45 +08:00
global CENTER_POINT, MAP_RATIO
2021-07-30 21:21:51 +08:00
map_path.mkdir(exist_ok=True, parents=True)
2021-08-04 15:19:45 +08:00
_map = map_path / "map.png"
2021-08-04 18:47:10 +08:00
if _map.exists() and os.path.getsize(_map) > 1024 * 1024 * 30:
2021-08-04 15:19:45 +08:00
_map.unlink()
2021-12-01 14:03:34 +08:00
try:
2022-02-19 20:03:40 +08:00
response = await AsyncHttpx.get(MAP_URL, timeout=10)
2021-12-01 14:03:34 +08:00
if response.status_code == 200:
data = response.json()
if data["message"] == "OK":
data = json.loads(data["data"]["info"]["detail"])
CENTER_POINT = (data["origin"][0], data["origin"][1])
if not _map.exists() or flag:
2021-12-01 14:03:34 +08:00
data = data["slices"]
idx = 0
w_len = len(data[0])
h_len = len(data)
for _map_data in data:
for _map in _map_data:
map_url = _map["url"]
await download_image(
map_url,
map_path / f"{idx}.png",
semaphore,
force_flag=flag,
)
BuildImage(
0, 0, background=f"{map_path}/{idx}.png", ratio=MAP_RATIO
).save()
idx += 1
w, h = BuildImage(0, 0, background=f"{map_path}/0.png").size
map_file = BuildImage(w * w_len, h * h_len, w, h, ratio=MAP_RATIO)
2021-12-01 14:03:34 +08:00
for i in range(idx):
img = BuildImage(0, 0, background=f"{map_path}/{i}.png")
await map_file.apaste(img)
2021-12-01 14:03:34 +08:00
map_file.save(f"{map_path}/map.png")
else:
logger.warning(f'获取原神地图失败 msg: {data["message"]}')
2021-05-20 19:34:27 +08:00
else:
2021-12-01 14:03:34 +08:00
logger.warning(f"获取原神地图失败 code{response.status_code}")
except (TimeoutError, ConnectTimeout):
logger.warning("下载原神地图数据超时....")
except Exception as e:
logger.error(f"下载原神地图数据失败 {type(e)}{e}")
2021-07-30 21:21:51 +08:00
# 下载资源类型数据
2021-11-23 21:44:59 +08:00
async def download_resource_type():
2021-07-30 21:21:51 +08:00
resource_type_file.parent.mkdir(parents=True, exist_ok=True)
2021-12-01 14:03:34 +08:00
try:
2022-02-19 20:03:40 +08:00
response = await AsyncHttpx.get(LABEL_URL, timeout=10)
2021-12-01 14:03:34 +08:00
if response.status_code == 200:
data = response.json()
if data["message"] == "OK":
data = data["data"]["tree"]
resource_data = {}
for x in data:
id_ = x["id"]
resource_data[id_] = x
with open(resource_type_file, "w", encoding="utf8") as f:
json.dump(resource_data, f, ensure_ascii=False, indent=4)
logger.info(f"更新原神资源类型成功...")
else:
logger.warning(f'获取原神资源类型失败 msg: {data["message"]}')
2021-07-30 21:21:51 +08:00
else:
2021-12-01 14:03:34 +08:00
logger.warning(f"获取原神资源类型失败 code{response.status_code}")
except (TimeoutError, ConnectTimeout):
logger.warning("下载原神资源类型数据超时....")
except Exception as e:
2022-05-22 18:08:06 +08:00
logger.error(f"载原神资源类型数据错误 {type(e)}{e}")
2021-07-30 21:21:51 +08:00
# 初始化资源图标
2022-05-22 18:08:06 +08:00
def gen_icon(icon: Path):
2021-12-16 11:16:28 +08:00
A = BuildImage(0, 0, background=f"{icon_path}/box.png")
B = BuildImage(0, 0, background=f"{icon_path}/box_alpha.png")
2022-05-22 18:08:06 +08:00
icon_img = BuildImage(115, 115, background=icon)
2021-07-30 21:21:51 +08:00
icon_img.circle()
B.paste(icon_img, (17, 10), True)
B.paste(A, alpha=True)
B.save(icon)
logger.info(f"生成图片成功 file{str(icon)}")
# 下载图片
async def download_image(
img_url: str,
2022-05-22 18:08:06 +08:00
path: Path,
2021-07-30 21:21:51 +08:00
semaphore: Semaphore,
gen_flag: bool = False,
force_flag: bool = False,
):
async with semaphore:
try:
2022-05-22 18:08:06 +08:00
if not path.exists() or not is_valid(path) or force_flag:
2022-02-19 20:03:40 +08:00
if await AsyncHttpx.download_file(img_url, path, timeout=10):
2021-11-23 21:44:59 +08:00
logger.info(f"下载原神资源图标:{img_url}")
if gen_flag:
gen_icon(path)
else:
logger.info(f"下载原神资源图标:{img_url} 失败,等待下次更新...")
2022-05-21 13:15:53 +08:00
except Exception as e:
logger.warning(f"原神图片错误..已删除,等待下次更新... file: {path} {type(e)}{e}")
2021-07-30 21:21:51 +08:00
if os.path.exists(path):
os.remove(path)
2021-08-04 15:19:45 +08:00
#
# def _get_point_ratio():
#