import os import shutil import sys import sysconfig from pathlib import Path import cx_Freeze is_64bits = sys.maxsize > 2 ** 32 folder = "exe.{platform}-{version}".format(platform=sysconfig.get_platform(), version=sysconfig.get_python_version()) buildfolder = Path("build", folder) sbuildfolder = str(buildfolder) libfolder = Path(buildfolder, "lib") library = Path(libfolder, "library.zip") print("Outputting to: " + sbuildfolder) icon = os.path.join("data", "icon.ico") if os.path.exists("X:/pw.txt"): print("Using signtool") with open("X:/pw.txt") as f: pw = f.read() signtool = r'signtool sign /f X:/_SITS_Zertifikat_.pfx /p ' + pw + r' /fd sha256 /tr http://timestamp.digicert.com/ ' else: signtool = None from hashlib import sha3_512 import base64 def _threaded_hash(filepath): hasher = sha3_512() hasher.update(open(filepath, "rb").read()) return base64.b85encode(hasher.digest()).decode() os.makedirs(buildfolder, exist_ok=True) def manifest_creation(folder): hashes = {} manifestpath = os.path.join(folder, "manifest.json") from concurrent.futures import ThreadPoolExecutor pool = ThreadPoolExecutor() for dirpath, dirnames, filenames in os.walk(folder): for filename in filenames: path = os.path.join(dirpath, filename) hashes[os.path.relpath(path, start=folder)] = pool.submit(_threaded_hash, path) import json from Utils import version_tuple manifest = {"buildtime": buildtime.isoformat(sep=" ", timespec="seconds"), "hashes": {path: hash.result() for path, hash in hashes.items()}, "version": version_tuple} json.dump(manifest, open(manifestpath, "wt"), indent=4) print("Created Manifest") scripts = {"LttPClient.py": "ArchipelagoLttPClient", "MultiMystery.py": "ArchipelagoMultiMystery", "MultiServer.py": "ArchipelagoServer", "Mystery.py": "ArchipelagoMystery", "LttPAdjuster.py": "ArchipelagoLttPAdjuster"} exes = [] for script, scriptname in scripts.items(): exes.append(cx_Freeze.Executable( script=script, target_name=scriptname + ("" if sys.platform == "linux" else ".exe"), icon=icon, )) import datetime buildtime = datetime.datetime.utcnow() cx_Freeze.setup( name="Archipelago", version=f"{buildtime.year}.{buildtime.month}.{buildtime.day}.{buildtime.hour}", description="Archipelago", executables=exes, options={ "build_exe": { "packages": ["websockets", "worlds"], "includes": [], "excludes": ["numpy", "Cython", "PySide2", "PIL", "pandas"], "zip_include_packages": ["*"], "zip_exclude_packages": ["worlds"], "include_files": [], "include_msvcr": True, "replace_paths": [("*", "")], "optimize": 2, "build_exe": buildfolder }, }, ) def installfile(path, keep_content=False): lbuildfolder = buildfolder print('copying', path, '->', lbuildfolder) if path.is_dir(): lbuildfolder /= path.name if lbuildfolder.is_dir() and not keep_content: shutil.rmtree(lbuildfolder) shutil.copytree(path, lbuildfolder, dirs_exist_ok=True) elif path.is_file(): shutil.copy(path, lbuildfolder) else: print('Warning,', path, 'not found') extra_data = ["LICENSE", "data", "EnemizerCLI", "host.yaml", "SNI", "meta.yaml"] for data in extra_data: installfile(Path(data)) os.makedirs(buildfolder / "Players", exist_ok=True) shutil.copyfile("playerSettings.yaml", buildfolder / "Players" / "weightedSettings.yaml") try: from maseya import z3pr except ImportError: print("Maseya Palette Shuffle not found, skipping data files.") else: # maseya Palette Shuffle exists and needs its data files print("Maseya Palette Shuffle found, including data files...") file = z3pr.__file__ installfile(Path(os.path.dirname(file)) / "data", keep_content=True) if signtool: for exe in exes: print(f"Signing {exe.target_name}") os.system(signtool + os.path.join(buildfolder, exe.target_name)) print(f"Signing SNI") os.system(signtool + os.path.join(buildfolder, "SNI", "SNI.exe")) alttpr_sprites_folder = buildfolder / "data" / "sprites" / "alttpr" for file in os.listdir(alttpr_sprites_folder): if file != ".gitignore": os.remove(alttpr_sprites_folder / file) manifest_creation(buildfolder) buildfolder = Path("build_factorio", folder) sbuildfolder = str(buildfolder) libfolder = Path(buildfolder, "lib") library = Path(libfolder, "library.zip") print("Outputting Factorio Client to: " + sbuildfolder) os.makedirs(buildfolder, exist_ok=True) exes = [ cx_Freeze.Executable( script="FactorioClient.py", target_name="ArchipelagoFactorioClient" + ("" if sys.platform == "linux" else ".exe"), icon=icon, base="Win32GUI" if sys.platform == "win32" else None )] import datetime buildtime = datetime.datetime.utcnow() cx_Freeze.setup( name="Archipelago Factorio Client", version=f"{buildtime.year}.{buildtime.month}.{buildtime.day}.{buildtime.hour}", description="Archipelago Factorio Client", executables=exes, options={ "build_exe": { "packages": ["websockets", "kivy", "worlds"], "includes": [], "excludes": ["numpy", "Cython", "PySide2", "PIL", "pandas"], "zip_include_packages": ["*"], "zip_exclude_packages": ["kivy", "worlds"], "include_files": [], "include_msvcr": True, "replace_paths": [("*", "")], "optimize": 2, "build_exe": buildfolder }, }, ) extra_data = ["LICENSE", "data", "host.yaml", "meta.yaml"] from kivy_deps import sdl2, glew for folder in sdl2.dep_bins+glew.dep_bins: shutil.copytree(folder, buildfolder, dirs_exist_ok=True) for data in extra_data: installfile(Path(data)) os.makedirs(buildfolder / "Players", exist_ok=True) shutil.copyfile("playerSettings.yaml", buildfolder / "Players" / "weightedSettings.yaml") if signtool: for exe in exes: print(f"Signing {exe.target_name}") os.system(signtool + os.path.join(buildfolder, exe.target_name)) alttpr_sprites_folder = buildfolder / "data" / "sprites" / "alttpr" for file in os.listdir(alttpr_sprites_folder): if file != ".gitignore": os.remove(alttpr_sprites_folder / file) manifest_creation(buildfolder)