-
Notifications
You must be signed in to change notification settings - Fork 23
/
multi_poly.py
148 lines (129 loc) · 6.68 KB
/
multi_poly.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
import shutil
from pathlib import Path
from typing import Callable, Optional, Tuple
from zipfile import ZIP_DEFLATED, ZipFile
from gtnh.assembler.downloader import get_asset_version_cache_location
from gtnh.assembler.generic_assembler import GenericAssembler
from gtnh.defs import JAVA_9_ARCHIVE_SUFFIX, MMC_ASSETS_DIR, MMC_PACK_INSTANCE, MMC_PACK_JSON, RELEASE_MMC_DIR, Side
from gtnh.models.gtnh_config import GTNHConfig
from gtnh.models.gtnh_release import GTNHRelease
from gtnh.models.gtnh_version import GTNHVersion
from gtnh.models.mod_info import GTNHModInfo
from gtnh.modpack_manager import GTNHModpackManager
class MMCAssembler(GenericAssembler):
"""
MMC assembler class. Allows for the assembling of mmc archives.
"""
def __init__(
self,
gtnh_modpack: GTNHModpackManager,
release: GTNHRelease,
task_progress_callback: Optional[Callable[[float, str], None]] = None,
global_progress_callback: Optional[Callable[[float, str], None]] = None,
changelog_path: Optional[Path] = None,
):
"""
Constructor of the MMCAssembler class.
:param gtnh_modpack: the modpack manager instance
:param release: the target release object
:param task_progress_callback: the callback to report the progress of the task
:param global_progress_callback: the callback to report the global progress
"""
GenericAssembler.__init__(
self,
gtnh_modpack=gtnh_modpack,
release=release,
task_progress_callback=task_progress_callback,
global_progress_callback=global_progress_callback,
changelog_path=changelog_path,
)
self.mmc_archive_root: Path = Path(f"GT New Horizons {self.release.version}")
self.mmc_modpack_files: Path = self.mmc_archive_root / ".minecraft"
self.mmc_modpack_mods: Path = self.mmc_modpack_files / "mods"
def add_mods(
self,
side: Side,
mods: list[tuple[GTNHModInfo, GTNHVersion]],
archive: ZipFile,
verbose: bool = False,
) -> None:
for mod, version in mods:
source_file: Path = get_asset_version_cache_location(mod, version)
archive_path: Path = self.mmc_modpack_mods / source_file.name
archive.write(source_file, arcname=archive_path)
for extra_asset in version.extra_assets:
if extra_asset.filename is not None and extra_asset.filename.endswith("multimc.zip"):
extra_asset_path: Path = get_asset_version_cache_location(mod, version, extra_asset.filename)
with ZipFile(extra_asset_path, "r", compression=ZIP_DEFLATED) as mmc_patches_zip:
for item in mmc_patches_zip.namelist():
with mmc_patches_zip.open(item, "r") as mmc_patch:
with archive.open(str(self.mmc_archive_root) + "/" + item, "w") as target:
shutil.copyfileobj(mmc_patch, target)
if self.task_progress_callback is not None:
self.task_progress_callback(
self.get_progress(), f"adding mod {mod.name} : version {version.version_tag} to the archive"
)
def add_config(
self, side: Side, config: Tuple[GTNHConfig, GTNHVersion], archive: ZipFile, verbose: bool = False
) -> None:
modpack_config: GTNHConfig
config_version: Optional[GTNHVersion]
modpack_config, config_version = config
config_file: Path = get_asset_version_cache_location(modpack_config, config_version)
with ZipFile(config_file, "r", compression=ZIP_DEFLATED) as config_zip:
for item in config_zip.namelist():
if item in self.exclusions[side]:
continue
with config_zip.open(item) as config_item:
with archive.open(
str(self.mmc_modpack_files) + "/" + item, "w"
) as target: # can't use Path for the whole
# path here as it strips leading / but those are used by
# zipfile to know if it's a file or a folder. If used here,
# Path objects will lead to the creation of empty files for
# every folder.
shutil.copyfileobj(config_item, target)
if self.task_progress_callback is not None:
self.task_progress_callback(self.get_progress(), f"adding {item} to the archive")
assert self.changelog_path
self.add_changelog(archive, arcname=self.mmc_modpack_files / self.changelog_path.name)
def get_archive_path(self, side: Side) -> Path:
suffix = "_Java_8"
if side.is_java9():
suffix = f"_{JAVA_9_ARCHIVE_SUFFIX}"
return RELEASE_MMC_DIR / f"GT_New_Horizons_{self.release.version}{suffix}.zip"
async def assemble(self, side: Side, verbose: bool = False) -> None:
if side not in {Side.CLIENT, Side.CLIENT_JAVA9}:
raise ValueError(f"Only valid sides are {Side.CLIENT.value}, got {side.value}")
# +1 for the metadata file
self.set_progress(
100
/ (
len(self.get_mods(side))
+ self.get_amount_of_files_in_config(side)
+ self.get_amount_of_files_in_locales()
+ 1
)
)
await GenericAssembler.assemble(self, side, verbose)
with ZipFile(self.get_archive_path(side), "a", compression=ZIP_DEFLATED) as archive:
self.add_localisation_files(archive, str(self.mmc_modpack_files.as_posix())) # otherwise file check fails
# on windows
self.add_mmc_meta_data(side)
def add_mmc_meta_data(self, side: Side) -> None:
"""
Method used to add additional meta data to the mmc archive.
:param side: client side
:return: None
"""
with ZipFile(self.get_archive_path(side), "a", compression=ZIP_DEFLATED) as archive:
if self.task_progress_callback is not None:
self.task_progress_callback(self.get_progress(), "adding archive's metadata to the archive")
if not side.is_java9():
archive.writestr(str(self.mmc_archive_root) + "/mmc-pack.json", MMC_PACK_JSON)
archive.writestr(
str(self.mmc_archive_root) + "/instance.cfg", MMC_PACK_INSTANCE.format(f"GTNH {self.release.version}")
)
with archive.open(str(self.mmc_archive_root) + "/gtnh_icon.png", "w") as target:
with open(MMC_ASSETS_DIR / "gtnh_icon.png", "rb") as icon:
shutil.copyfileobj(icon, target)