repo: upload files
This commit is contained in:
commit
7c60dc18a6
160
.gitignore
vendored
Normal file
160
.gitignore
vendored
Normal file
@ -0,0 +1,160 @@
|
|||||||
|
# Byte-compiled / optimized / DLL files
|
||||||
|
__pycache__/
|
||||||
|
*.py[cod]
|
||||||
|
*$py.class
|
||||||
|
|
||||||
|
# C extensions
|
||||||
|
*.so
|
||||||
|
|
||||||
|
# Distribution / packaging
|
||||||
|
.Python
|
||||||
|
build/
|
||||||
|
develop-eggs/
|
||||||
|
dist/
|
||||||
|
downloads/
|
||||||
|
eggs/
|
||||||
|
.eggs/
|
||||||
|
lib/
|
||||||
|
lib64/
|
||||||
|
parts/
|
||||||
|
sdist/
|
||||||
|
var/
|
||||||
|
wheels/
|
||||||
|
share/python-wheels/
|
||||||
|
*.egg-info/
|
||||||
|
.installed.cfg
|
||||||
|
*.egg
|
||||||
|
MANIFEST
|
||||||
|
|
||||||
|
# PyInstaller
|
||||||
|
# Usually these files are written by a python script from a template
|
||||||
|
# before PyInstaller builds the exe, so as to inject date/other infos into it.
|
||||||
|
*.manifest
|
||||||
|
*.spec
|
||||||
|
|
||||||
|
# Installer logs
|
||||||
|
pip-log.txt
|
||||||
|
pip-delete-this-directory.txt
|
||||||
|
|
||||||
|
# Unit test / coverage reports
|
||||||
|
htmlcov/
|
||||||
|
.tox/
|
||||||
|
.nox/
|
||||||
|
.coverage
|
||||||
|
.coverage.*
|
||||||
|
.cache
|
||||||
|
nosetests.xml
|
||||||
|
coverage.xml
|
||||||
|
*.cover
|
||||||
|
*.py,cover
|
||||||
|
.hypothesis/
|
||||||
|
.pytest_cache/
|
||||||
|
cover/
|
||||||
|
|
||||||
|
# Translations
|
||||||
|
*.mo
|
||||||
|
*.pot
|
||||||
|
|
||||||
|
# Django stuff:
|
||||||
|
*.log
|
||||||
|
local_settings.py
|
||||||
|
db.sqlite3
|
||||||
|
db.sqlite3-journal
|
||||||
|
|
||||||
|
# Flask stuff:
|
||||||
|
instance/
|
||||||
|
.webassets-cache
|
||||||
|
|
||||||
|
# Scrapy stuff:
|
||||||
|
.scrapy
|
||||||
|
|
||||||
|
# Sphinx documentation
|
||||||
|
docs/_build/
|
||||||
|
|
||||||
|
# PyBuilder
|
||||||
|
.pybuilder/
|
||||||
|
target/
|
||||||
|
|
||||||
|
# Jupyter Notebook
|
||||||
|
.ipynb_checkpoints
|
||||||
|
|
||||||
|
# IPython
|
||||||
|
profile_default/
|
||||||
|
ipython_config.py
|
||||||
|
|
||||||
|
# pyenv
|
||||||
|
# For a library or package, you might want to ignore these files since the code is
|
||||||
|
# intended to run in multiple environments; otherwise, check them in:
|
||||||
|
# .python-version
|
||||||
|
|
||||||
|
# pipenv
|
||||||
|
# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
|
||||||
|
# However, in case of collaboration, if having platform-specific dependencies or dependencies
|
||||||
|
# having no cross-platform support, pipenv may install dependencies that don't work, or not
|
||||||
|
# install all needed dependencies.
|
||||||
|
#Pipfile.lock
|
||||||
|
|
||||||
|
# poetry
|
||||||
|
# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control.
|
||||||
|
# This is especially recommended for binary packages to ensure reproducibility, and is more
|
||||||
|
# commonly ignored for libraries.
|
||||||
|
# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control
|
||||||
|
#poetry.lock
|
||||||
|
|
||||||
|
# pdm
|
||||||
|
# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control.
|
||||||
|
#pdm.lock
|
||||||
|
# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it
|
||||||
|
# in version control.
|
||||||
|
# https://pdm.fming.dev/#use-with-ide
|
||||||
|
.pdm.toml
|
||||||
|
|
||||||
|
# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm
|
||||||
|
__pypackages__/
|
||||||
|
|
||||||
|
# Celery stuff
|
||||||
|
celerybeat-schedule
|
||||||
|
celerybeat.pid
|
||||||
|
|
||||||
|
# SageMath parsed files
|
||||||
|
*.sage.py
|
||||||
|
|
||||||
|
# Environments
|
||||||
|
.env
|
||||||
|
.venv
|
||||||
|
env/
|
||||||
|
venv/
|
||||||
|
ENV/
|
||||||
|
env.bak/
|
||||||
|
venv.bak/
|
||||||
|
|
||||||
|
# Spyder project settings
|
||||||
|
.spyderproject
|
||||||
|
.spyproject
|
||||||
|
|
||||||
|
# Rope project settings
|
||||||
|
.ropeproject
|
||||||
|
|
||||||
|
# mkdocs documentation
|
||||||
|
/site
|
||||||
|
|
||||||
|
# mypy
|
||||||
|
.mypy_cache/
|
||||||
|
.dmypy.json
|
||||||
|
dmypy.json
|
||||||
|
|
||||||
|
# Pyre type checker
|
||||||
|
.pyre/
|
||||||
|
|
||||||
|
# pytype static type analyzer
|
||||||
|
.pytype/
|
||||||
|
|
||||||
|
# Cython debug symbols
|
||||||
|
cython_debug/
|
||||||
|
|
||||||
|
# PyCharm
|
||||||
|
# JetBrains specific template is maintained in a separate JetBrains.gitignore that can
|
||||||
|
# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
|
||||||
|
# and can be added to the global gitignore or merged into this file. For a more nuclear
|
||||||
|
# option (not recommended) you can uncomment the following to ignore the entire idea folder.
|
||||||
|
#.idea/
|
21
LICENSE
Normal file
21
LICENSE
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2023 tretrauit
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE.
|
32
README.md
Normal file
32
README.md
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
# Vollerei
|
||||||
|
|
||||||
|
[![Code style: black](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/psf/black)
|
||||||
|
|
||||||
|
An open-source launcher for anime games
|
||||||
|
|
||||||
|
## Why?
|
||||||
|
|
||||||
|
I've done [worthless-launcher](https://tretrauit.gitlab.io/worthless-launcher) for an open-world anime game,
|
||||||
|
since I want to support other anime games and the launcher code is very messy, this launcher was made.
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
+ Nothing, I haven't written any code yet.
|
||||||
|
|
||||||
|
### Notes
|
||||||
|
|
||||||
|
This launcher tries to mimic the official launcher behaviour as much as possible but if a ban appears, I will
|
||||||
|
not be responsible for it. (*Turn-based game* have a ban wave already, see AAGL discord for more info)
|
||||||
|
|
||||||
|
## Alternatives
|
||||||
|
|
||||||
|
This launcher focuses on the API and CLI, for GUI-based launcher you may want to check out:
|
||||||
|
|
||||||
|
+ [An Anime Game Launcher](https://aagl.launcher.moe/) - That famous launcher for an open-world anime game.
|
||||||
|
+ [Yaagl]
|
||||||
|
+ [Honkers launcher](https://github.com/an-anime-team/honkers-launcher) - Another launcher for an anime game.
|
||||||
|
+ [Honkers Railway](https://github.com/an-anime-team/the-honkers-railway-launcher) - A launcher for a turn-based anime game.
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
[MIT](./LICENSE)
|
89
poetry.lock
generated
Normal file
89
poetry.lock
generated
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
# This file is automatically @generated by Poetry 1.5.0 and should not be changed by hand.
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "colorama"
|
||||||
|
version = "0.4.6"
|
||||||
|
description = "Cross-platform colored terminal text."
|
||||||
|
optional = false
|
||||||
|
python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7"
|
||||||
|
files = [
|
||||||
|
{file = "colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6"},
|
||||||
|
{file = "colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44"},
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "iniconfig"
|
||||||
|
version = "2.0.0"
|
||||||
|
description = "brain-dead simple config-ini parsing"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=3.7"
|
||||||
|
files = [
|
||||||
|
{file = "iniconfig-2.0.0-py3-none-any.whl", hash = "sha256:b6a85871a79d2e3b22d2d1b94ac2824226a63c6b741c88f7ae975f18b6778374"},
|
||||||
|
{file = "iniconfig-2.0.0.tar.gz", hash = "sha256:2d91e135bf72d31a410b17c16da610a82cb55f6b0477d1a902134b24a455b8b3"},
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "packaging"
|
||||||
|
version = "23.1"
|
||||||
|
description = "Core utilities for Python packages"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=3.7"
|
||||||
|
files = [
|
||||||
|
{file = "packaging-23.1-py3-none-any.whl", hash = "sha256:994793af429502c4ea2ebf6bf664629d07c1a9fe974af92966e4b8d2df7edc61"},
|
||||||
|
{file = "packaging-23.1.tar.gz", hash = "sha256:a392980d2b6cffa644431898be54b0045151319d1e7ec34f0cfed48767dd334f"},
|
||||||
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "platformdirs"
|
||||||
|
version = "3.5.1"
|
||||||
|
description = "A small Python package for determining appropriate platform-specific dirs, e.g. a \"user data dir\"."
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=3.7"
|
||||||
|
files = [
|
||||||
|
{file = "platformdirs-3.5.1-py3-none-any.whl", hash = "sha256:e2378146f1964972c03c085bb5662ae80b2b8c06226c54b2ff4aa9483e8a13a5"},
|
||||||
|
{file = "platformdirs-3.5.1.tar.gz", hash = "sha256:412dae91f52a6f84830f39a8078cecd0e866cb72294a5c66808e74d5e88d251f"},
|
||||||
|
]
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
docs = ["furo (>=2023.3.27)", "proselint (>=0.13)", "sphinx (>=6.2.1)", "sphinx-autodoc-typehints (>=1.23,!=1.23.4)"]
|
||||||
|
test = ["appdirs (==1.4.4)", "covdefaults (>=2.3)", "pytest (>=7.3.1)", "pytest-cov (>=4)", "pytest-mock (>=3.10)"]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "pluggy"
|
||||||
|
version = "1.0.0"
|
||||||
|
description = "plugin and hook calling mechanisms for python"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=3.6"
|
||||||
|
files = [
|
||||||
|
{file = "pluggy-1.0.0-py2.py3-none-any.whl", hash = "sha256:74134bbf457f031a36d68416e1509f34bd5ccc019f0bcc952c7b909d06b37bd3"},
|
||||||
|
{file = "pluggy-1.0.0.tar.gz", hash = "sha256:4224373bacce55f955a878bf9cfa763c1e360858e330072059e10bad68531159"},
|
||||||
|
]
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
dev = ["pre-commit", "tox"]
|
||||||
|
testing = ["pytest", "pytest-benchmark"]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "pytest"
|
||||||
|
version = "7.3.1"
|
||||||
|
description = "pytest: simple powerful testing with Python"
|
||||||
|
optional = false
|
||||||
|
python-versions = ">=3.7"
|
||||||
|
files = [
|
||||||
|
{file = "pytest-7.3.1-py3-none-any.whl", hash = "sha256:3799fa815351fea3a5e96ac7e503a96fa51cc9942c3753cda7651b93c1cfa362"},
|
||||||
|
{file = "pytest-7.3.1.tar.gz", hash = "sha256:434afafd78b1d78ed0addf160ad2b77a30d35d4bdf8af234fe621919d9ed15e3"},
|
||||||
|
]
|
||||||
|
|
||||||
|
[package.dependencies]
|
||||||
|
colorama = {version = "*", markers = "sys_platform == \"win32\""}
|
||||||
|
iniconfig = "*"
|
||||||
|
packaging = "*"
|
||||||
|
pluggy = ">=0.12,<2.0"
|
||||||
|
|
||||||
|
[package.extras]
|
||||||
|
testing = ["argcomplete", "attrs (>=19.2.0)", "hypothesis (>=3.56)", "mock", "nose", "pygments (>=2.7.2)", "requests", "xmlschema"]
|
||||||
|
|
||||||
|
[metadata]
|
||||||
|
lock-version = "2.0"
|
||||||
|
python-versions = "^3.11"
|
||||||
|
content-hash = "b34ef1fca1c3d1052c3cd98113215a31309abdebdc436b62c81257507b0cb1ed"
|
18
pyproject.toml
Normal file
18
pyproject.toml
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
[tool.poetry]
|
||||||
|
name = "vollerei"
|
||||||
|
version = "0.1.0"
|
||||||
|
description = "An open-source launcher for anime games"
|
||||||
|
authors = ["tretrauit <tretrauit@gmail.com>"]
|
||||||
|
license = "MIT"
|
||||||
|
readme = "README.md"
|
||||||
|
|
||||||
|
[tool.poetry.dependencies]
|
||||||
|
python = "^3.11"
|
||||||
|
platformdirs = "^3.5.1"
|
||||||
|
|
||||||
|
[tool.poetry.group.dev.dependencies]
|
||||||
|
pytest = "^7.3.1"
|
||||||
|
|
||||||
|
[build-system]
|
||||||
|
requires = ["poetry-core"]
|
||||||
|
build-backend = "poetry.core.masonry.api"
|
0
vollerei/__init__.py
Normal file
0
vollerei/__init__.py
Normal file
0
vollerei/abc/__init__.py
Normal file
0
vollerei/abc/__init__.py
Normal file
0
vollerei/abc/launcher/__init__.py
Normal file
0
vollerei/abc/launcher/__init__.py
Normal file
93
vollerei/abc/launcher/game.py
Normal file
93
vollerei/abc/launcher/game.py
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
from abc import ABC, abstractmethod
|
||||||
|
from os import PathLike
|
||||||
|
|
||||||
|
|
||||||
|
class GameABC(ABC):
|
||||||
|
"""
|
||||||
|
Manages the game installation
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, path: PathLike = None):
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def is_installed(self) -> bool:
|
||||||
|
"""
|
||||||
|
Check if the game is installed
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def install_game(self, game_path: PathLike = None):
|
||||||
|
"""
|
||||||
|
Install the game
|
||||||
|
|
||||||
|
If path is not specified then it'll use self.path, if that is
|
||||||
|
not specified too then it'll raise an exception.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
game_path (PathLike, optional): Path to install the game to.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
None
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def install_game_from_archive(
|
||||||
|
self, archive: PathLike, game_path: PathLike = None
|
||||||
|
) -> None:
|
||||||
|
"""
|
||||||
|
Install the game from an archive
|
||||||
|
|
||||||
|
If path is not specified then it'll use self.path, if that is
|
||||||
|
not specified too then it'll raise an exception.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
archive (PathLike): Path to the archive.
|
||||||
|
game_path (PathLike, optional): Path to install the game to.
|
||||||
|
"""
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def install_update_from_archive(
|
||||||
|
self, archive: PathLike, game_path: PathLike = None
|
||||||
|
) -> None:
|
||||||
|
"""
|
||||||
|
Install the update from an archive
|
||||||
|
|
||||||
|
Args:
|
||||||
|
archive (PathLike): Path to the archive
|
||||||
|
game_path (PathLike, optional): Path to the game. Defaults to None.
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def get_version(self) -> tuple[int, int, int]:
|
||||||
|
"""
|
||||||
|
Get the game version
|
||||||
|
|
||||||
|
If the game is not installed, it'll return (0, 0, 0).
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def get_update(self):
|
||||||
|
"""
|
||||||
|
Get the game update
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def get_voiceover_update(self, language: str):
|
||||||
|
"""
|
||||||
|
Get the voiceover update
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def get_channel(self):
|
||||||
|
"""
|
||||||
|
Get the game channel
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
0
vollerei/abc/launcher/interface.py
Normal file
0
vollerei/abc/launcher/interface.py
Normal file
16
vollerei/abc/patcher.py
Normal file
16
vollerei/abc/patcher.py
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
from abc import ABC, abstractmethod
|
||||||
|
|
||||||
|
from vollerei.abc.launcher.game import GameABC
|
||||||
|
|
||||||
|
|
||||||
|
class PatcherABC(ABC):
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def patch_game(self, game: GameABC):
|
||||||
|
pass
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def unpatch_game(self, game: GameABC):
|
||||||
|
pass
|
24
vollerei/constants.py
Normal file
24
vollerei/constants.py
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
from platformdirs import PlatformDirs
|
||||||
|
|
||||||
|
# Common
|
||||||
|
telemetry_hosts = [
|
||||||
|
# Global
|
||||||
|
"log-upload-os.hoyoverse.com",
|
||||||
|
"sg-public-data-api.hoyoverse.com",
|
||||||
|
# China
|
||||||
|
"dump.gamesafe.qq.com",
|
||||||
|
"log-upload.mihoyo.com",
|
||||||
|
"public-data-api.mihoyo.com",
|
||||||
|
]
|
||||||
|
|
||||||
|
# HSR
|
||||||
|
astra_repo = "https://notabug.org/mkrsym1/astra"
|
||||||
|
jadeite_repo = "https://codeberg.org/mkrsym1/jadeite/"
|
||||||
|
hsr_latest_version = (1, 1, 0)
|
||||||
|
|
||||||
|
base_dirs = PlatformDirs("vollerei", "tretrauit", roaming=True)
|
||||||
|
tools_data_path = base_dirs.site_data_path.joinpath("tools")
|
||||||
|
tools_cache_path = base_dirs.site_cache_path.joinpath("tools")
|
||||||
|
tools_cache_path.mkdir(parents=True, exist_ok=True)
|
||||||
|
launcher_cache_path = base_dirs.site_cache_path.joinpath("launcher")
|
||||||
|
launcher_cache_path.mkdir(parents=True, exist_ok=True)
|
4
vollerei/exceptions/__init__.py
Normal file
4
vollerei/exceptions/__init__.py
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
class VollereiError(Exception):
|
||||||
|
"""Base class for exceptions in this module."""
|
||||||
|
|
||||||
|
pass
|
13
vollerei/exceptions/game.py
Normal file
13
vollerei/exceptions/game.py
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
from vollerei.exceptions import VollereiError
|
||||||
|
|
||||||
|
|
||||||
|
class GameError(VollereiError):
|
||||||
|
"""Base class for exceptions in related to the game installation."""
|
||||||
|
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class GameNotInstalledError(GameError):
|
||||||
|
"""Exception raised when the game is not installed."""
|
||||||
|
|
||||||
|
pass
|
13
vollerei/exceptions/patcher.py
Normal file
13
vollerei/exceptions/patcher.py
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
from vollerei.exceptions import VollereiError
|
||||||
|
|
||||||
|
|
||||||
|
class PatcherError(VollereiError):
|
||||||
|
"""Base class for exceptions in related to the patcher."""
|
||||||
|
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class VersionNotSupportedError(PatcherError):
|
||||||
|
"""Exception raised when the game version is not supported."""
|
||||||
|
|
||||||
|
pass
|
0
vollerei/genshin/__init__.py
Normal file
0
vollerei/genshin/__init__.py
Normal file
0
vollerei/genshin/launcher/__init__.py
Normal file
0
vollerei/genshin/launcher/__init__.py
Normal file
0
vollerei/genshin/launcher/interface.py
Normal file
0
vollerei/genshin/launcher/interface.py
Normal file
0
vollerei/genshin/patcher.py
Normal file
0
vollerei/genshin/patcher.py
Normal file
0
vollerei/honkai/__init__.py
Normal file
0
vollerei/honkai/__init__.py
Normal file
0
vollerei/honkai/launcher/__init__.py
Normal file
0
vollerei/honkai/launcher/__init__.py
Normal file
0
vollerei/honkai/launcher/interface.py
Normal file
0
vollerei/honkai/launcher/interface.py
Normal file
0
vollerei/honkai/patcher.py
Normal file
0
vollerei/honkai/patcher.py
Normal file
0
vollerei/hsr/__init__.py
Normal file
0
vollerei/hsr/__init__.py
Normal file
0
vollerei/hsr/launcher/__init__.py
Normal file
0
vollerei/hsr/launcher/__init__.py
Normal file
17
vollerei/hsr/launcher/game.py
Normal file
17
vollerei/hsr/launcher/game.py
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
from os import PathLike
|
||||||
|
from pathlib import Path
|
||||||
|
from vollerei.abc.launcher.game import GameABC
|
||||||
|
|
||||||
|
|
||||||
|
class Game(GameABC):
|
||||||
|
def __init__(self, path: PathLike = None):
|
||||||
|
self.path: Path | None = Path(path) if path else None
|
||||||
|
|
||||||
|
def is_installed(self) -> bool:
|
||||||
|
if self.path is None:
|
||||||
|
return False
|
||||||
|
if (
|
||||||
|
not self.path.joinpath("StarRail.exe").exists()
|
||||||
|
or not self.path.joinpath("StarRailBase.dll").exists()
|
||||||
|
):
|
||||||
|
return False
|
0
vollerei/hsr/launcher/interface.py
Normal file
0
vollerei/hsr/launcher/interface.py
Normal file
61
vollerei/hsr/patcher.py
Normal file
61
vollerei/hsr/patcher.py
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
from vollerei.abc.patcher import PatcherABC
|
||||||
|
from vollerei.exceptions.game import GameNotInstalledError
|
||||||
|
from vollerei.exceptions.patcher import VersionNotSupportedError
|
||||||
|
from vollerei.hsr.launcher.game import Game
|
||||||
|
from vollerei.utils.git import Git
|
||||||
|
from vollerei.constants import tools_data_path, astra_repo, jadeite_repo
|
||||||
|
from enum import Enum
|
||||||
|
|
||||||
|
|
||||||
|
class PatchType(Enum):
|
||||||
|
"""
|
||||||
|
Patch type
|
||||||
|
|
||||||
|
Astra: The old patch which patch the game directly (not recommended).
|
||||||
|
Jadeite: The new patch which patch the game in memory by DLL injection.
|
||||||
|
"""
|
||||||
|
|
||||||
|
Astra: int = 0
|
||||||
|
Jadeite: int = 1
|
||||||
|
|
||||||
|
|
||||||
|
class Patcher(PatcherABC):
|
||||||
|
def __init__(self, patch_type: PatchType = PatchType.Jadeite):
|
||||||
|
self._patch_type: PatchType = patch_type
|
||||||
|
self._path = tools_data_path.joinpath("patcher")
|
||||||
|
self._git = Git()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def patch_type(self) -> PatchType:
|
||||||
|
return self._patch_type
|
||||||
|
|
||||||
|
@patch_type.setter
|
||||||
|
def patch_type(self, value: PatchType):
|
||||||
|
self._patch_type = value
|
||||||
|
|
||||||
|
def _update_astra(self):
|
||||||
|
self._git.pull_or_clone(astra_repo, self._path.joinpath("astra"))
|
||||||
|
|
||||||
|
def _update_jadeite(self):
|
||||||
|
self._git.pull_or_clone(jadeite_repo, self._path.joinpath("jadeite"))
|
||||||
|
|
||||||
|
def update_patch(self):
|
||||||
|
match self._patch_type:
|
||||||
|
case PatchType.Astra:
|
||||||
|
self._update_astra()
|
||||||
|
case PatchType.Jadeite:
|
||||||
|
self._update_jadeite()
|
||||||
|
|
||||||
|
def _patch_astra(self, game: Game):
|
||||||
|
if game.get_version() != (1, 0, 5):
|
||||||
|
raise VersionNotSupportedError(
|
||||||
|
"Only version 1.0.5 is supported by Astra patch."
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _patch_jadeite(self, game: Game):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def patch_game(self, game: Game):
|
||||||
|
if not game.is_installed():
|
||||||
|
raise GameNotInstalledError("Game is not installed")
|
45
vollerei/utils/git/__init__.py
Normal file
45
vollerei/utils/git/__init__.py
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
import subprocess
|
||||||
|
from pathlib import Path
|
||||||
|
from shutil import which
|
||||||
|
from vollerei.utils.git.exceptions import GitCloneError, GitNotInstalled
|
||||||
|
|
||||||
|
|
||||||
|
class Git:
|
||||||
|
"""
|
||||||
|
Quick wrapper around git binary
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def check_git():
|
||||||
|
"""
|
||||||
|
Check for git installation, if not found raise GitNotInstalled
|
||||||
|
"""
|
||||||
|
if not which("git"):
|
||||||
|
raise GitNotInstalled("git is not installed")
|
||||||
|
|
||||||
|
def pull_or_clone(self, url: str, path: str = None) -> None:
|
||||||
|
self.check_git()
|
||||||
|
"""
|
||||||
|
Pulls or clones a git repository
|
||||||
|
|
||||||
|
If the repository already exists and the url matches, it'll be pulled.
|
||||||
|
"""
|
||||||
|
if path is None:
|
||||||
|
path = Path.cwd().joinpath(Path(url).stem)
|
||||||
|
try:
|
||||||
|
origin_url = subprocess.check_output(
|
||||||
|
["git", "config", "--get", "remote.origin.url"], cwd=path
|
||||||
|
).decode()
|
||||||
|
if origin_url != url:
|
||||||
|
raise subprocess.CalledProcessError
|
||||||
|
subprocess.check_call(["git", "pull"], cwd=path)
|
||||||
|
except subprocess.CalledProcessError:
|
||||||
|
try:
|
||||||
|
subprocess.check_call(["git", "clone", url, path])
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
raise GitCloneError(
|
||||||
|
f"Failed to clone or update repository {url} to {path}"
|
||||||
|
) from e
|
22
vollerei/utils/git/exceptions.py
Normal file
22
vollerei/utils/git/exceptions.py
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
class GitError(Exception):
|
||||||
|
"""Base class for git errors"""
|
||||||
|
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class GitCloneError(GitError):
|
||||||
|
"""Raised when git clone fails"""
|
||||||
|
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class GitPullError(GitError):
|
||||||
|
"""Raised when git pull fails"""
|
||||||
|
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class GitNotInstalled(GitError):
|
||||||
|
"""Raised when git is not installed"""
|
||||||
|
|
||||||
|
pass
|
0
vollerei/utils/hdiffpatch.py
Normal file
0
vollerei/utils/hdiffpatch.py
Normal file
81
vollerei/utils/xdelta3/__init__.py
Normal file
81
vollerei/utils/xdelta3/__init__.py
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
import platform
|
||||||
|
import subprocess
|
||||||
|
import requests
|
||||||
|
from os import PathLike
|
||||||
|
from zipfile import ZipFile
|
||||||
|
from shutil import which
|
||||||
|
from vollerei.constants import tools_cache_path
|
||||||
|
from vollerei.utils.xdelta3.exceptions import (
|
||||||
|
Xdelta3NotInstalledError,
|
||||||
|
Xdelta3PatchError,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class Xdelta3:
|
||||||
|
"""
|
||||||
|
Quick wrapper around xdelta3 binary
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
self._xdelta3_path = tools_cache_path.joinpath("xdelta3")
|
||||||
|
self._xdelta3_path.mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
|
def _get_binary(self, recurse=None) -> str:
|
||||||
|
if which("xdelta3"):
|
||||||
|
return "xdelta3"
|
||||||
|
if platform.system() == "Windows":
|
||||||
|
for path in self._xdelta3_path.glob("*.exe"):
|
||||||
|
return path
|
||||||
|
if recurse is None:
|
||||||
|
recurse = 3
|
||||||
|
elif recurse == 0:
|
||||||
|
raise Xdelta3NotInstalledError(
|
||||||
|
"xdelta3 is not installed and can't be automatically installed"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
recurse -= 1
|
||||||
|
self.download()
|
||||||
|
return self.get_binary(recurse=recurse)
|
||||||
|
raise Xdelta3NotInstalledError("xdelta3 is not installed")
|
||||||
|
|
||||||
|
def get_binary(self) -> str:
|
||||||
|
"""
|
||||||
|
Get xdelta3 binary
|
||||||
|
"""
|
||||||
|
return self._get_binary()
|
||||||
|
|
||||||
|
def download(self):
|
||||||
|
"""
|
||||||
|
Download xdelta3 binary
|
||||||
|
"""
|
||||||
|
url = None
|
||||||
|
if platform.system() != "Windows":
|
||||||
|
raise NotImplementedError(
|
||||||
|
"xdelta3 binary is not available for this platform, please install it manually" # noqa: E501
|
||||||
|
)
|
||||||
|
match platform.machine():
|
||||||
|
case "AMD64":
|
||||||
|
url = "https://github.com/jmacd/xdelta-gpl/releases/download/v3.1.0/xdelta3-3.1.0-x86_64.exe.zip"
|
||||||
|
case "i386":
|
||||||
|
url = "https://github.com/jmacd/xdelta-gpl/releases/download/v3.1.0/xdelta3-3.1.0-i686.exe.zip"
|
||||||
|
case "i686":
|
||||||
|
url = "https://github.com/jmacd/xdelta-gpl/releases/download/v3.1.0/xdelta3-3.1.0-i686.exe.zip"
|
||||||
|
with requests.get(url, stream=True) as r:
|
||||||
|
with open(self._xdelta3_path.joinpath("xdelta3.zip"), "wb") as f:
|
||||||
|
for chunk in r.iter_content(chunk_size=32768):
|
||||||
|
f.write(chunk)
|
||||||
|
with ZipFile(self._xdelta3_path.joinpath("xdelta3.zip")) as z:
|
||||||
|
z.extractall(self._xdelta3_path)
|
||||||
|
|
||||||
|
def patch_file(self, patch: PathLike, target: PathLike, output: PathLike):
|
||||||
|
"""
|
||||||
|
Patch a file
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
subprocess.check_call(
|
||||||
|
[self.get_binary(), "-d", "-s", patch, target, output]
|
||||||
|
)
|
||||||
|
except subprocess.CalledProcessError as e:
|
||||||
|
raise Xdelta3PatchError(
|
||||||
|
f"xdelta3 failed with exit code {e.returncode}"
|
||||||
|
) from e
|
16
vollerei/utils/xdelta3/exceptions.py
Normal file
16
vollerei/utils/xdelta3/exceptions.py
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
class Xdelta3Error(Exception):
|
||||||
|
"""Base class for xdelta3 errors"""
|
||||||
|
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class Xdelta3NotInstalledError(Xdelta3Error):
|
||||||
|
"""Raised when xdelta3 is not installed"""
|
||||||
|
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class Xdelta3PatchError(Xdelta3Error):
|
||||||
|
"""Raised when xdelta3 patch fails"""
|
||||||
|
|
||||||
|
pass
|
Loading…
Reference in New Issue
Block a user