From 47181e076325b142136e4c42137537156b510df4 Mon Sep 17 00:00:00 2001 From: ashlen Date: Wed, 9 Oct 2024 09:47:46 +0200 Subject: [PATCH] Support tw gamedata bundles (#9) Updated the unity.py version and added support for tw gamedata with a new schema repo. --- .gitignore | 5 +++- arkprts/assets/bundle.py | 52 +++++++++++++++++++++------------------- pyproject.toml | 2 +- requirements.txt | 2 +- setup.py | 6 ++--- 5 files changed, 36 insertions(+), 31 deletions(-) diff --git a/.gitignore b/.gitignore index 341e9a2..b41b55a 100644 --- a/.gitignore +++ b/.gitignore @@ -61,4 +61,7 @@ venv.bak/ # mypy .mypy_cache/ .dmypy.json -dmypy.json \ No newline at end of file +dmypy.json + +# custom +flatc diff --git a/arkprts/assets/bundle.py b/arkprts/assets/bundle.py index 4832ac1..f91f5da 100644 --- a/arkprts/assets/bundle.py +++ b/arkprts/assets/bundle.py @@ -11,7 +11,6 @@ import io import json import logging -import os import pathlib import re import shlex @@ -33,7 +32,7 @@ UnityPyAsset = typing.Any UnityPyObject = typing.Any -UPDATED_FBS = {"cn": False, "yostar": False} +UPDATED_FBS = {"cn": False, "yostar": False, "tw": False} def asset_path_to_server_filename(path: str) -> str: @@ -109,7 +108,7 @@ def run_flatbuffers( "--defaults-json", "--unknown-json", "--raw-binary", - # "--no-warnings", + "--no-warnings", "--force-empty", ] result = subprocess.run(args, stdout=subprocess.PIPE, stderr=subprocess.PIPE, check=False) # noqa: S603, UP022 @@ -124,16 +123,14 @@ def run_flatbuffers( return pathlib.Path(output_directory) / (pathlib.Path(fbs_path).stem + ".json") -def resolve_fbs_schema_directory(server: typing.Literal["cn", "yostar"]) -> pathlib.Path: +def resolve_fbs_schema_directory(server: typing.Literal["cn", "yostar", "tw"]) -> pathlib.Path: """Resolve the flatbuffers schema directory.""" - path = os.environ.get(f"FLATBUFFERS_SCHEMA_DIR_{server.upper()}") - if path: - return pathlib.Path(path) + if server == "tw": + return netn.APPDATA_DIR / "ArknightsFlatbuffers" / "tw" core_path = netn.APPDATA_DIR / "ArknightsFBS" core_path.mkdir(parents=True, exist_ok=True) path = core_path / server / "OpenArknightsFBS" / "FBS" - os.environ[f"FLATBUFFERS_SCHEMA_DIR_{server.upper()}"] = str(path) return path @@ -147,6 +144,14 @@ async def update_fbs_schema(*, force: bool = False) -> None: directory = resolve_fbs_schema_directory(server).parent # pyright: ignore[reportArgumentType] await git.download_repository("MooncellWiki/OpenArknightsFBS", directory, branch=branch, force=force) + if not UPDATED_FBS["tw"] or force: + UPDATED_FBS["tw"] = True + await git.download_repository( + "ArknightsAssets/ArknightsFlatbuffers", + netn.APPDATA_DIR / "ArknightsFlatbuffers", + force=force, + ) + def recursively_collapse_keys(obj: typing.Any) -> typing.Any: """Recursively collapse arknights flatc dictionaries.""" @@ -176,14 +181,13 @@ def decrypt_fbs_file( if rsa: data = data[128:] - tempdir = netn.TEMP_DIR / "ArknightsFBS" + tempdir = netn.TEMP_DIR / "ArknightsFBS" / server tempdir.mkdir(parents=True, exist_ok=True) fbs_path = tempdir / (table_name + ".bytes") fbs_path.write_bytes(data) - fbs_schema_path = resolve_fbs_schema_directory(server="cn" if server in ("cn", "bili") else "yostar") / ( - table_name + ".fbs" - ) + ser = "cn" if server in ("cn", "bili") else "tw" if server == "tw" else "yostar" + fbs_schema_path = resolve_fbs_schema_directory(ser) / (table_name + ".fbs") output_directory = tempdir / "output" output_path = run_flatbuffers(fbs_path, fbs_schema_path, output_directory) @@ -242,40 +246,38 @@ def find_ab_assets( """Yield relative paths and data for a unity asset.""" for container, obj in asset.container.items(): if obj.type.name == "TextAsset": + data = obj.read() + script, name = data.m_Script.encode("utf-8", "surrogateescape"), data.m_Name + if match := re.match(DYNP + r"(.+\.txt)", container): - data = obj.read() - yield (match[1], data.script) + yield (match[1], script) continue if match := re.match(DYNP + r"(gamedata/.+?\.json)", container): - data = obj.read() - yield (match[1], normalize_json(bytes(data.script), lenient=not normalize)) + yield (match[1], normalize_json(bytes(script), lenient=not normalize)) continue if match := re.match(DYNP + r"(gamedata/.+?)\.lua\.bytes", container): - data = obj.read() - text = decrypt_aes_text(data.script) + text = decrypt_aes_text(script) yield (match[1] + ".lua", text) continue if match := re.match(DYNP + r"(gamedata/levels/(?:obt|activities)/.+?)\.bytes", container): - data = obj.read() try: - text = normalize_json(bytes(data.script)[128:], lenient=not normalize) + text = normalize_json(bytes(script)[128:], lenient=not normalize) except UnboundLocalError: # effectively bson's "type not recognized" error - text = decrypt_fbs_file(data.script, "prts___levels", server=server) + text = decrypt_fbs_file(script, "prts___levels", server=server) yield (match[1] + ".json", text) continue if match := re.match(DYNP + r"(gamedata/.+?)(?:[a-fA-F0-9]{6})?\.bytes", container): - data = obj.read() # the only rsa-less file is ~~global~~ tw's enemy_database text = decrypt_arknights_text( - data.script, - name=data.name, - rsa=data.name != "enemy_database", + script, + name=name, + rsa=name != "enemy_database", server=server, normalize=normalize, ) diff --git a/pyproject.toml b/pyproject.toml index e51d1bf..6db0365 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,7 +1,7 @@ [project] name = "arkprts" requires-python = ">=3.9" -version = "0.3.10" +version = "0.3.11" dynamic = [ "dependencies", "description", diff --git a/requirements.txt b/requirements.txt index 468b4a5..0bb6a3c 100644 --- a/requirements.txt +++ b/requirements.txt @@ -3,5 +3,5 @@ pydantic==2.* rsa pycryptodome -UnityPy +UnityPy>=1.20 bson diff --git a/setup.py b/setup.py index fd165ee..230839b 100644 --- a/setup.py +++ b/setup.py @@ -5,7 +5,7 @@ setup( name="arkprts", - version="0.3.10", + version="0.3.11", description="Arknights python wrapper.", url="https://github.com/thesadru/arkprts", packages=find_packages(exclude=["tests", "tests.*"]), @@ -13,10 +13,10 @@ package_data={"arkprts": ["py.typed"]}, install_requires=["aiohttp", "pydantic==2.*"], extras_require={ - "all": ["rsa", "pycryptodome", "UnityPy", "bson"], + "all": ["rsa", "pycryptodome", "UnityPy>=1.20", "bson"], "rsa": ["rsa"], "aes": ["pycryptodome"], - "assets": ["UnityPy", "pycryptodome", "bson"], + "assets": ["UnityPy>=1.20", "pycryptodome", "bson"], }, long_description=pathlib.Path("README.md").read_text(encoding="utf-8"), long_description_content_type="text/markdown",