| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  | import bsdiff4 | 
					
						
							|  |  |  | import yaml | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | import lzma | 
					
						
							| 
									
										
										
										
											2020-03-07 00:07:45 +01:00
										 |  |  | import hashlib | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  | from typing import Tuple | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | import Utils | 
					
						
							| 
									
										
										
										
											2020-03-07 00:30:14 +01:00
										 |  |  | from Rom import JAP10HASH, read_rom | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | base_rom_bytes = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_base_rom_bytes() -> bytes: | 
					
						
							|  |  |  |     global base_rom_bytes | 
					
						
							|  |  |  |     if not base_rom_bytes: | 
					
						
							| 
									
										
										
										
											2020-03-15 19:32:00 +01:00
										 |  |  |         options = Utils.get_options() | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  |         file_name = options["general_options"]["rom_file"] | 
					
						
							| 
									
										
										
										
											2020-03-15 19:32:00 +01:00
										 |  |  |         if not os.path.exists(file_name): | 
					
						
							|  |  |  |             file_name = Utils.local_path(file_name) | 
					
						
							| 
									
										
										
										
											2020-03-07 00:30:14 +01:00
										 |  |  |         base_rom_bytes = bytes(read_rom(open(file_name, "rb"))) | 
					
						
							| 
									
										
										
										
											2020-03-07 00:07:45 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         basemd5 = hashlib.md5() | 
					
						
							|  |  |  |         basemd5.update(base_rom_bytes) | 
					
						
							|  |  |  |         if JAP10HASH != basemd5.hexdigest(): | 
					
						
							| 
									
										
										
										
											2020-03-08 02:18:55 +01:00
										 |  |  |             raise Exception('Supplied Base Rom does not match known MD5 for JAP(1.0) release. ' | 
					
						
							|  |  |  |                             'Get the correct game and version, then dump it') | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  |     return base_rom_bytes | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def generate_patch(rom: bytes, metadata=None) -> bytes: | 
					
						
							|  |  |  |     if metadata is None: | 
					
						
							|  |  |  |         metadata = {} | 
					
						
							|  |  |  |     patch = bsdiff4.diff(get_base_rom_bytes(), rom) | 
					
						
							|  |  |  |     patch = yaml.dump({"meta": metadata, | 
					
						
							|  |  |  |                        "patch": patch}) | 
					
						
							| 
									
										
										
										
											2020-03-22 20:04:30 +01:00
										 |  |  |     return patch.encode(encoding="utf-8-sig") | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def create_patch_file(rom_file_to_patch: str, server: str = "") -> str: | 
					
						
							|  |  |  |     bytes = generate_patch(load_bytes(rom_file_to_patch), | 
					
						
							|  |  |  |                            { | 
					
						
							|  |  |  |                                "server": server})  # allow immediate connection to server in multiworld. Empty string otherwise | 
					
						
							| 
									
										
										
										
											2020-03-06 00:54:13 +01:00
										 |  |  |     target = os.path.splitext(rom_file_to_patch)[0] + ".bmbp" | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  |     write_lzma(bytes, target) | 
					
						
							|  |  |  |     return target | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  | def create_rom_file(patch_file) -> Tuple[dict, str]: | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  |     data = Utils.parse_yaml(lzma.decompress(load_bytes(patch_file)).decode("utf-8-sig")) | 
					
						
							|  |  |  |     patched_data = bsdiff4.patch(get_base_rom_bytes(), data["patch"]) | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  |     target = os.path.splitext(patch_file)[0] + ".sfc" | 
					
						
							|  |  |  |     with open(target, "wb") as f: | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  |         f.write(patched_data) | 
					
						
							| 
									
										
										
										
											2020-03-10 00:38:29 +01:00
										 |  |  |     return data["meta"], target | 
					
						
							| 
									
										
										
										
											2020-03-06 00:48:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def load_bytes(path: str): | 
					
						
							|  |  |  |     with open(path, "rb") as f: | 
					
						
							|  |  |  |         return f.read() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def write_lzma(data: bytes, path: str): | 
					
						
							|  |  |  |     with lzma.LZMAFile(path, 'wb') as f: | 
					
						
							|  |  |  |         f.write(data) | 
					
						
							| 
									
										
										
										
											2020-03-17 19:16:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     ipv4 = Utils.get_public_ipv4() | 
					
						
							|  |  |  |     import sys | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for rom in sys.argv: | 
					
						
							|  |  |  |         if rom.endswith(".sfc"): | 
					
						
							|  |  |  |             print(f"Creating patch for {rom}") | 
					
						
							|  |  |  |             result = create_patch_file(rom, ipv4) | 
					
						
							|  |  |  |             print(f"Created patch {result}") |